1
0
Fork 0
mirror of https://github.com/pry/pry.git synced 2022-11-09 12:35:05 -05:00
pry--pry/spec/command_set_spec.rb

665 lines
19 KiB
Ruby
Raw Normal View History

require_relative 'helper'
describe Pry::CommandSet do
before do
@set = Pry::CommandSet.new do
import Pry::Commands
end
2011-12-31 06:10:23 -05:00
@ctx = {
:target => binding,
2014-02-15 08:29:59 -05:00
:command_set => @set,
:pry_instance => Pry.new(output: StringIO.new)
2011-12-31 06:10:23 -05:00
}
end
2013-09-22 05:21:51 -04:00
describe "[]=" do
it "removes a command from the command set" do
2015-01-22 16:52:20 -05:00
@set["help"].should_not eq nil
2013-09-22 05:21:51 -04:00
@set["help"] = nil
2015-01-22 16:52:20 -05:00
@set["help"].should eq nil
expect { @set.run_command(TOPLEVEL_BINDING, "help") }.to raise_error Pry::NoCommandError
2013-09-22 05:21:51 -04:00
end
it "replaces a command" do
old_help = @set["help"]
@set["help"] = @set["pry-version"]
2015-01-22 16:52:20 -05:00
@set["help"].should_not eq old_help
2013-09-22 05:21:51 -04:00
end
it "rebinds the command with key" do
@set["help-1"] = @set["help"]
2015-01-22 16:52:20 -05:00
@set["help-1"].match.should eq "help-1"
2013-09-22 05:21:51 -04:00
end
2013-09-22 05:39:10 -04:00
it "raises a TypeError when command is not a subclass of Pry::Command" do
expect { @set["help"] = "hello" }.to raise_error TypeError
2013-09-22 05:39:10 -04:00
end
2013-09-22 05:21:51 -04:00
end
it 'should call the block used for the command when it is called' do
run = false
@set.command 'foo' do
run = true
end
2011-09-25 02:52:07 -04:00
@set.run_command @ctx, 'foo'
2015-01-22 16:52:20 -05:00
run.should eq true
end
it 'should pass arguments of the command to the block' do
2014-09-12 11:55:59 -04:00
expect { |probe|
@set.command('foo', &probe)
@set.run_command(@ctx, 'foo', 1, 2, 3)
}.to yield_with_args(1, 2, 3)
end
2011-12-31 06:10:23 -05:00
it 'should use the first argument as context' do
2014-09-12 12:40:37 -04:00
inside = inner_scope do |probe|
@set.command('foo', &probe)
2011-09-25 02:52:07 -04:00
2014-09-12 12:40:37 -04:00
@set.run_command @ctx, 'foo'
end
2014-09-12 12:40:37 -04:00
expect(inside.context).to eq(@ctx)
end
it 'should raise an error when calling an undefined command' do
@set.command('foo') {}
expect { @set.run_command @ctx, 'bar' }.to raise_error Pry::NoCommandError
end
it 'should be able to remove its own commands' do
@set.command('foo') {}
@set.delete 'foo'
expect { @set.run_command @ctx, 'foo' }.to raise_error Pry::NoCommandError
end
it 'should be able to remove its own commands, by listing name' do
@set.command(/^foo1/, 'desc', :listing => 'foo') {}
@set.delete 'foo'
expect { @set.run_command @ctx, /^foo1/ }.to raise_error Pry::NoCommandError
end
it 'should be able to import some commands from other sets' do
run = false
2011-05-07 01:32:05 -04:00
other_set = Pry::CommandSet.new do
command('foo') { run = true }
command('bar') {}
end
@set.import_from(other_set, 'foo')
2011-09-25 02:52:07 -04:00
@set.run_command @ctx, 'foo'
2015-01-22 16:52:20 -05:00
run.should eq true
expect { @set.run_command @ctx, 'bar' }.to raise_error Pry::NoCommandError
end
it 'should return command set after import' do
run = false
other_set = Pry::CommandSet.new do
command('foo') { run = true }
command('bar') {}
end
2015-01-22 16:52:20 -05:00
@set.import(other_set).should eq @set
end
it 'should return command set after import_from' do
run = false
other_set = Pry::CommandSet.new do
command('foo') { run = true }
command('bar') {}
end
2015-01-22 16:52:20 -05:00
@set.import_from(other_set, 'foo').should eq @set
end
it 'should be able to import some commands from other sets using listing name' do
run = false
other_set = Pry::CommandSet.new do
command(/^foo1/, 'desc', :listing => 'foo') { run = true }
end
@set.import_from(other_set, 'foo')
@set.run_command @ctx, /^foo1/
2015-01-22 16:52:20 -05:00
run.should eq true
end
it 'should be able to import a whole set' do
run = []
2011-05-07 01:32:05 -04:00
other_set = Pry::CommandSet.new do
command('foo') { run << true }
command('bar') { run << true }
end
@set.import other_set
2011-09-25 02:52:07 -04:00
@set.run_command @ctx, 'foo'
@set.run_command @ctx, 'bar'
2015-01-22 16:52:20 -05:00
run.should eq [true, true]
end
it 'should be able to import sets at creation' do
run = false
@set.command('foo') { run = true }
2011-09-25 02:52:07 -04:00
Pry::CommandSet.new(@set).run_command @ctx, 'foo'
2015-01-22 16:52:20 -05:00
run.should eq true
end
it 'should set the descriptions of commands' do
@set.command('foo', 'some stuff') {}
2015-01-22 16:52:20 -05:00
@set['foo'].description.should eq 'some stuff'
end
describe "aliases" do
it 'should be able to alias command' do
run = false
@set.command('foo', 'stuff') { run = true }
@set.alias_command 'bar', 'foo'
2015-01-22 16:52:20 -05:00
@set['bar'].match.should eq 'bar'
@set['bar'].description.should eq 'Alias for `foo`'
@set.run_command @ctx, 'bar'
2015-01-22 16:52:20 -05:00
run.should eq true
end
it "should be able to alias command with command_prefix" do
run = false
2014-02-15 08:29:59 -05:00
begin
@set.command('owl', 'stuff') { run = true }
@set.alias_command 'owlet', 'owl'
Pry.config.command_prefix = '%'
2015-01-22 16:52:20 -05:00
@set['%owlet'].match.should eq 'owlet'
@set['%owlet'].description.should eq 'Alias for `owl`'
2014-02-15 08:29:59 -05:00
@set.run_command @ctx, 'owlet'
2015-01-22 16:52:20 -05:00
run.should eq true
2014-02-15 08:29:59 -05:00
ensure
Pry.config.command_prefix = ''
end
end
it 'should inherit options from original command' do
run = false
@set.command('foo', 'stuff', :shellwords => true, :interpolate => false) { run = true }
@set.alias_command 'bar', 'foo'
2015-01-22 16:52:20 -05:00
@set['bar'].options[:shellwords].should eq @set['foo'].options[:shellwords]
@set['bar'].options[:interpolate].should eq @set['foo'].options[:interpolate]
# however some options should not be inherited
2015-01-22 16:52:20 -05:00
@set['bar'].options[:listing].should_not eq @set['foo'].options[:listing]
@set['bar'].options[:listing].should eq "bar"
end
it 'should be able to specify alias\'s description when aliasing' do
run = false
@set.command('foo', 'stuff') { run = true }
@set.alias_command 'bar', 'foo', :desc => "tobina"
2015-01-22 16:52:20 -05:00
@set['bar'].match.should eq 'bar'
@set['bar'].description.should eq "tobina"
@set.run_command @ctx, 'bar'
2015-01-22 16:52:20 -05:00
run.should eq true
end
it "should be able to alias a command by its invocation line" do
run = false
@set.command(/^foo1/, 'stuff', :listing => 'foo') { run = true }
@set.alias_command 'bar', 'foo1'
2015-01-22 16:52:20 -05:00
@set['bar'].match.should eq 'bar'
@set['bar'].description.should eq 'Alias for `foo1`'
@set.run_command @ctx, 'bar'
2015-01-22 16:52:20 -05:00
run.should eq true
end
it "should be able to specify options when creating alias" do
run = false
@set.command(/^foo1/, 'stuff', :listing => 'foo') { run = true }
2015-01-22 16:52:20 -05:00
@set.alias_command(/^b.r/, 'foo1', :listing => "bar")
@set.to_hash[/^b.r/].options[:listing].should eq "bar"
end
it "should set description to default if description parameter is nil" do
run = false
@set.command(/^foo1/, 'stuff', :listing => 'foo') { run = true }
@set.alias_command "bar", 'foo1'
2015-01-22 16:52:20 -05:00
@set["bar"].description.should eq "Alias for `foo1`"
end
end
it 'should be able to change the descriptions of commands' do
@set.command('foo', 'bar') {}
@set.desc 'foo', 'baz'
2015-01-22 16:52:20 -05:00
@set['foo'].description.should eq 'baz'
end
it 'should get the descriptions of commands' do
@set.command('foo', 'bar') {}
2015-01-22 16:52:20 -05:00
@set.desc('foo').should eq 'bar'
end
it 'should get the descriptions of commands, by listing' do
@set.command(/^foo1/, 'bar', :listing => 'foo') {}
2015-01-22 16:52:20 -05:00
@set.desc('foo').should eq 'bar'
end
it 'should return Pry::Command::VOID_VALUE for commands by default' do
@set.command('foo') { 3 }
2015-01-22 16:52:20 -05:00
@set.run_command(@ctx, 'foo').should eq Pry::Command::VOID_VALUE
end
it 'should be able to keep return values' do
@set.command('foo', '', :keep_retval => true) { 3 }
2015-01-22 16:52:20 -05:00
@set.run_command(@ctx, 'foo').should eq 3
end
it 'should be able to keep return values, even if return value is nil' do
@set.command('foo', '', :keep_retval => true) { nil }
2015-01-22 16:52:20 -05:00
@set.run_command(@ctx, 'foo').should eq nil
end
it 'should be able to have its own helpers' do
@set.command('foo') { my_helper }
@set.helpers { def my_helper; end }
2011-09-25 02:52:07 -04:00
@set.run_command(@ctx, 'foo')
Pry::Command.subclass('foo', '', {}, Module.new)
.new({:target => binding})
.should_not(respond_to :my_helper)
end
it 'should not recreate a new helper module when helpers is called' do
@set.command('foo') do
my_helper
my_other_helper
end
@set.helpers do
def my_helper; end
end
@set.helpers do
def my_other_helper; end
end
2011-09-25 02:52:07 -04:00
@set.run_command(@ctx, 'foo')
end
it 'should import helpers from imported sets' do
2011-05-07 01:32:05 -04:00
imported_set = Pry::CommandSet.new do
helpers do
def imported_helper_method; end
end
end
@set.import imported_set
@set.command('foo') { imported_helper_method }
2011-09-25 02:52:07 -04:00
@set.run_command(@ctx, 'foo')
end
it 'should import helpers even if only some commands are imported' do
2011-05-07 01:32:05 -04:00
imported_set = Pry::CommandSet.new do
helpers do
def imported_helper_method; end
end
command('bar') {}
end
@set.import_from imported_set, 'bar'
@set.command('foo') { imported_helper_method }
2011-09-25 02:52:07 -04:00
@set.run_command(@ctx, 'foo')
end
it 'should provide a :listing for a command that defaults to its name' do
@set.command 'foo', "" do;end
2015-01-22 16:52:20 -05:00
@set['foo'].options[:listing].should eq 'foo'
end
it 'should provide a :listing for a command that differs from its name' do
@set.command 'foo', "", :listing => 'bar' do;end
2015-01-22 16:52:20 -05:00
@set['foo'].options[:listing].should eq 'bar'
end
2011-07-27 14:42:36 -04:00
it "should provide a 'help' command" do
2011-12-31 06:10:23 -05:00
@ctx[:command_set] = @set
@ctx[:output] = StringIO.new
2011-07-27 14:42:36 -04:00
expect { @set.run_command(@ctx, 'help') }.to_not raise_error
2011-07-27 14:42:36 -04:00
end
describe "renaming a command" do
it 'should be able to rename and run a command' do
run = false
@set.command('foo') { run = true }
@set.rename_command('bar', 'foo')
@set.run_command(@ctx, 'bar')
2015-01-22 16:52:20 -05:00
run.should eq true
end
it 'should accept listing name when renaming a command' do
run = false
@set.command('foo', "", :listing => 'love') { run = true }
@set.rename_command('bar', 'love')
@set.run_command(@ctx, 'bar')
2015-01-22 16:52:20 -05:00
run.should eq true
end
it 'should raise exception trying to rename non-existent command' do
expect { @set.rename_command('bar', 'foo') }.to raise_error ArgumentError
end
it 'should make old command name inaccessible' do
@set.command('foo') { }
@set.rename_command('bar', 'foo')
expect { @set.run_command(@ctx, 'foo') }.to raise_error Pry::NoCommandError
end
it 'should be able to pass in options when renaming command' do
desc = "hello"
listing = "bing"
@set.command('foo') { }
@set.rename_command('bar', 'foo', :description => desc, :listing => listing, :keep_retval => true)
2015-01-22 16:52:20 -05:00
@set['bar'].description.should eq desc
@set['bar'].options[:listing].should eq listing
@set['bar'].options[:keep_retval].should eq true
end
end
describe "command decorators - before_command and after_command" do
describe "before_command" do
it 'should be called before the original command' do
foo = []
@set.command('foo') { foo << 1 }
@set.before_command('foo') { foo << 2 }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
foo.should eq [2, 1]
end
it 'should be called before the original command, using listing name' do
foo = []
@set.command(/^foo1/, '', :listing => 'foo') { foo << 1 }
@set.before_command('foo') { foo << 2 }
@set.run_command(@ctx, /^foo1/)
2015-01-22 16:52:20 -05:00
foo.should eq [2, 1]
end
it 'should share the context with the original command' do
2011-12-31 06:10:23 -05:00
@ctx[:target] = "test target string".__binding__
before_val = nil
orig_val = nil
@set.command('foo') { orig_val = target }
@set.before_command('foo') { before_val = target }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
before_val.should eq @ctx[:target]
orig_val.should eq @ctx[:target]
end
it 'should work when applied multiple times' do
foo = []
@set.command('foo') { foo << 1 }
@set.before_command('foo') { foo << 2 }
@set.before_command('foo') { foo << 3 }
@set.before_command('foo') { foo << 4 }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
foo.should eq [4, 3, 2, 1]
end
end
describe "after_command" do
it 'should be called after the original command' do
foo = []
@set.command('foo') { foo << 1 }
@set.after_command('foo') { foo << 2 }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
foo.should eq [1, 2]
end
it 'should be called after the original command, using listing name' do
foo = []
@set.command(/^foo1/, '', :listing => 'foo') { foo << 1 }
@set.after_command('foo') { foo << 2 }
@set.run_command(@ctx, /^foo1/)
2015-01-22 16:52:20 -05:00
foo.should eq [1, 2]
end
it 'should share the context with the original command' do
2011-12-31 06:10:23 -05:00
@ctx[:target] = "test target string".__binding__
after_val = nil
orig_val = nil
@set.command('foo') { orig_val = target }
@set.after_command('foo') { after_val = target }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
after_val.should eq @ctx[:target]
orig_val.should eq @ctx[:target]
end
it 'should determine the return value for the command' do
@set.command('foo', 'bar', :keep_retval => true) { 1 }
@set.after_command('foo') { 2 }
2015-01-22 16:52:20 -05:00
@set.run_command(@ctx, 'foo').should eq 2
end
it 'should work when applied multiple times' do
foo = []
@set.command('foo') { foo << 1 }
@set.after_command('foo') { foo << 2 }
@set.after_command('foo') { foo << 3 }
@set.after_command('foo') { foo << 4 }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
foo.should eq [1, 2, 3, 4]
end
end
describe "before_command and after_command" do
it 'should work when combining both before_command and after_command' do
foo = []
@set.command('foo') { foo << 1 }
@set.after_command('foo') { foo << 2 }
@set.before_command('foo') { foo << 3 }
@set.run_command(@ctx, 'foo')
2015-01-22 16:52:20 -05:00
foo.should eq [3, 1, 2]
end
end
end
describe 'find_command' do
it 'should find commands with the right string' do
cmd = @set.command('rincewind'){ }
2015-01-22 16:52:20 -05:00
@set.find_command('rincewind').should eq cmd
end
it 'should not find commands with spaces before' do
2015-01-22 16:52:20 -05:00
@set.command('luggage'){ }
@set.find_command(' luggage').should eq nil
end
it 'should find commands with arguments after' do
cmd = @set.command('vetinari'){ }
2015-01-22 16:52:20 -05:00
@set.find_command('vetinari --knock 3').should eq cmd
end
it 'should find commands with names containing spaces' do
cmd = @set.command('nobby nobbs'){ }
2015-01-22 16:52:20 -05:00
@set.find_command('nobby nobbs --steal petty-cash').should eq cmd
end
it 'should find command defined by regex' do
cmd = @set.command(/(capt|captain) vimes/i){ }
2015-01-22 16:52:20 -05:00
@set.find_command('Capt Vimes').should eq cmd
end
it 'should find commands defined by regex with arguments' do
cmd = @set.command(/(cpl|corporal) Carrot/i){ }
2015-01-22 16:52:20 -05:00
@set.find_command('cpl carrot --write-home').should eq cmd
end
it 'should not find commands by listing' do
2015-01-22 16:52:20 -05:00
@set.command(/werewol(f|ve)s?/, 'only once a month', :listing => "angua"){ }
@set.find_command('angua').should eq nil
end
it 'should not find commands without command_prefix' do
begin
Pry.config.command_prefix = '%'
2015-01-22 16:52:20 -05:00
@set.command('detritus'){ }
@set.find_command('detritus').should eq nil
ensure
Pry.config.command_prefix = ''
end
end
it "should find commands that don't use the prefix" do
begin
Pry.config.command_prefix = '%'
cmd = @set.command('colon', 'Sergeant Fred', :use_prefix => false){ }
2015-01-22 16:52:20 -05:00
@set.find_command('colon').should eq cmd
ensure
Pry.config.command_prefix = ''
end
end
it "should find the command that has the longest match" do
2015-01-22 16:52:20 -05:00
@set.command(/\.(.*)/){ }
cmd2 = @set.command(/\.\|\|(.*)/){ }
2015-01-22 16:52:20 -05:00
@set.find_command('.||').should eq cmd2
end
it "should find the command that has the longest name" do
2015-01-22 16:52:20 -05:00
@set.command(/\.(.*)/){ }
cmd2 = @set.command('.||'){ }
2015-01-22 16:52:20 -05:00
@set.find_command('.||').should eq cmd2
end
end
describe '.valid_command?' do
it 'should be true for commands that can be found' do
2015-01-22 16:52:20 -05:00
@set.command('archchancellor')
@set.valid_command?('archchancellor of_the?(:University)').should eq true
end
it 'should be false for commands that can\'' do
2015-01-22 16:52:20 -05:00
@set.valid_command?('def monkey(ape)').should eq false
end
it 'should not cause argument interpolation' do
2015-01-22 16:52:20 -05:00
@set.command('hello')
expect { @set.valid_command?('hello #{raise "futz"}') }.to_not raise_error
end
end
describe '.process_line' do
it 'should return Result.new(false) if there is no matching command' do
result = @set.process_line('1 + 42')
2015-01-22 16:52:20 -05:00
result.command?.should eq false
result.void_command?.should eq false
result.retval.should eq nil
end
it 'should return Result.new(true, VOID) if the command is not keep_retval' do
@set.create_command('mrs-cake') do
def process; 42; end
end
result = @set.process_line('mrs-cake')
2015-01-22 16:52:20 -05:00
result.command?.should eq true
result.void_command?.should eq true
result.retval.should eq Pry::Command::VOID_VALUE
end
it 'should return Result.new(true, retval) if the command is keep_retval' do
@set.create_command('magrat', 'the maiden', :keep_retval => true) do
def process; 42; end
end
result = @set.process_line('magrat')
2015-01-22 16:52:20 -05:00
result.command?.should eq true
result.void_command?.should eq false
result.retval.should eq 42
end
it 'should pass through context' do
ctx = {
:eval_string => "bloomers",
:pry_instance => Object.new,
:output => StringIO.new,
:target => binding
}
2014-09-12 12:40:37 -04:00
inside = inner_scope do |probe|
@set.create_command('agnes') do
define_method(:process, &probe)
end
2014-09-12 12:40:37 -04:00
@set.process_line('agnes', ctx)
end
2014-09-12 12:40:37 -04:00
expect(inside.eval_string).to eq(ctx[:eval_string])
expect(inside.output).to eq(ctx[:output])
expect(inside.target).to eq(ctx[:target])
expect(inside._pry_).to eq(ctx[:pry_instance])
end
it 'should add command_set to context' do
2014-09-12 12:40:37 -04:00
inside = inner_scope do |probe|
@set.create_command(/nann+y ogg+/) do
define_method(:process, &probe)
end
2014-09-12 12:40:37 -04:00
@set.process_line('nannnnnny oggggg')
end
2014-09-12 12:40:37 -04:00
expect(inside.command_set).to eq(@set)
end
end
2012-08-21 01:12:03 -04:00
2012-09-06 01:39:29 -04:00
if defined?(Bond)
describe '.complete' do
it "should list all command names" do
@set.create_command('susan'){ }
@set.complete('sus').should.include 'susan '
2012-09-06 01:39:29 -04:00
end
2012-08-21 01:12:03 -04:00
2012-09-06 01:39:29 -04:00
it "should delegate to commands" do
@set.create_command('susan'){ def complete(search); ['--foo']; end }
2015-01-22 16:52:20 -05:00
@set.complete('susan ').should eq ['--foo']
2012-09-06 01:39:29 -04:00
end
2012-08-21 01:12:03 -04:00
end
end
end