summaryrefslogtreecommitdiff
path: root/home-config/zsh/zsh-autosuggestions/spec
diff options
context:
space:
mode:
Diffstat (limited to 'home-config/zsh/zsh-autosuggestions/spec')
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/async_spec.rb70
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/auto_cd_spec.rb14
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/bracketed_paste_magic_spec.rb43
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/client_zpty_spec.rb14
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/glob_subst_spec.rb12
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/rebound_bracket_spec.rb13
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/vi_mode_spec.rb80
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/wrapped_widget_spec.rb39
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/integrations/zle_input_stack_spec.rb24
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/kill_ring_spec.rb23
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/line_init_spec.rb17
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/multi_line_spec.rb8
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/options/buffer_max_size_spec.rb30
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/options/highlight_style_spec.rb7
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/options/original_widget_prefix_spec.rb7
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/options/strategy_spec.rb55
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/options/widget_lists_spec.rb121
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/spec_helper.rb54
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/strategies/completion_spec.rb72
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/strategies/history_spec.rb23
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/strategies/match_prev_cmd_spec.rb34
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/strategies/special_characters_helper.rb75
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/terminal_session.rb99
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/widgets/disable_spec.rb19
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/widgets/enable_spec.rb42
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/widgets/fetch_spec.rb24
-rw-r--r--home-config/zsh/zsh-autosuggestions/spec/widgets/toggle_spec.rb26
27 files changed, 1045 insertions, 0 deletions
diff --git a/home-config/zsh/zsh-autosuggestions/spec/async_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/async_spec.rb
new file mode 100644
index 0000000..0af7232
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/async_spec.rb
@@ -0,0 +1,70 @@
+context 'with asynchronous suggestions enabled' do
+ let(:options) { ["ZSH_AUTOSUGGEST_USE_ASYNC="] }
+
+ describe '`up-line-or-beginning-search`' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('autoload -U up-line-or-beginning-search').
+ run_command('zle -N up-line-or-beginning-search').
+ send_string('bindkey "').
+ send_keys('C-v').send_keys('up').
+ send_string('" up-line-or-beginning-search').
+ send_keys('enter')
+ end
+ end
+
+ it 'should show previous history entries' do
+ with_history(
+ 'echo foo',
+ 'echo bar',
+ 'echo baz'
+ ) do
+ session.clear_screen
+ 3.times { session.send_keys('up') }
+ wait_for { session.content }.to eq("echo foo")
+ end
+ end
+ end
+
+ describe '`copy-earlier-word`' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('autoload -Uz copy-earlier-word').
+ run_command('zle -N copy-earlier-word').
+ run_command('bindkey "^N" copy-earlier-word')
+ end
+ end
+
+ it 'should cycle through previous words in the buffer' do
+ session.clear_screen
+ session.send_string('foo bar baz')
+ sleep 0.5
+ session.send_keys('C-n')
+ wait_for { session.content }.to eq('foo bar bazbaz')
+ session.send_keys('C-n')
+ wait_for { session.content }.to eq('foo bar bazbar')
+ session.send_keys('C-n')
+ wait_for { session.content }.to eq('foo bar bazfoo')
+ end
+ end
+
+ describe 'pressing ^C after fetching a suggestion' do
+ before do
+ skip 'Workaround does not work below v5.0.8' if session.zsh_version < Gem::Version.new('5.0.8')
+ end
+
+ it 'terminates the prompt and begins a new one' do
+ session.send_keys('e')
+ sleep 0.5
+ session.send_keys('C-c')
+ sleep 0.5
+ session.send_keys('echo')
+
+ wait_for { session.content }.to eq("e\necho")
+ end
+ end
+end
+
+
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/auto_cd_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/auto_cd_spec.rb
new file mode 100644
index 0000000..94bd24b
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/auto_cd_spec.rb
@@ -0,0 +1,14 @@
+describe 'with `AUTO_CD` option set' do
+ let(:after_sourcing) do
+ -> {
+ session.run_command('setopt AUTO_CD')
+ session.run_command('autoload compinit && compinit')
+ }
+ end
+
+ it 'directory names are still completed' do
+ session.send_string('sr')
+ session.send_keys('C-i')
+ wait_for { session.content }.to eq('src/')
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/bracketed_paste_magic_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/bracketed_paste_magic_spec.rb
new file mode 100644
index 0000000..41ff267
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/bracketed_paste_magic_spec.rb
@@ -0,0 +1,43 @@
+describe 'pasting using bracketed-paste-magic' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('autoload -Uz bracketed-paste-magic').
+ run_command('zle -N bracketed-paste bracketed-paste-magic')
+ end
+ end
+
+ context 'with suggestions disabled while pasting' do
+ before do
+ session.
+ run_command('bpm_init() { zle autosuggest-disable }').
+ run_command('bpm_finish() { zle autosuggest-enable }').
+ run_command('zstyle :bracketed-paste-magic paste-init bpm_init').
+ run_command('zstyle :bracketed-paste-magic paste-finish bpm_finish')
+ end
+
+ it 'does not show an incorrect suggestion' do
+ with_history('echo hello') do
+ session.paste_string("echo #{'a' * 60}")
+ sleep 1
+ expect(session.content).to eq("echo #{'a' * 60}")
+ end
+ end
+ end
+
+ context 'with `bracketed-paste` added to the list of widgets that clear the suggestion' do
+ let(:options) { ['ZSH_AUTOSUGGEST_CLEAR_WIDGETS+=(bracketed-paste)'] }
+
+ it 'does not retain an old suggestion' do
+ with_history ('echo foo') do
+ session.send_string('echo ')
+ wait_for { session.content }.to eq('echo foo')
+ session.paste_string('bar')
+ wait_for { session.content }.to eq('echo bar')
+ session.send_keys('C-a') # Any cursor movement works
+ sleep 1
+ expect(session.content).to eq('echo bar')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/client_zpty_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/client_zpty_spec.rb
new file mode 100644
index 0000000..b8abb37
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/client_zpty_spec.rb
@@ -0,0 +1,14 @@
+describe 'a running zpty command' do
+ let(:before_sourcing) { -> { session.run_command('zmodload zsh/zpty && zpty -b kitty cat') } }
+
+ context 'when using `completion` strategy' do
+ let(:options) { ["ZSH_AUTOSUGGEST_STRATEGY=completion"] }
+
+ it 'is not affected' do
+ session.send_keys('a').send_keys('C-h')
+ session.run_command('zpty -t kitty; echo $?')
+
+ wait_for { session.content }.to end_with("\n0")
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/glob_subst_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/glob_subst_spec.rb
new file mode 100644
index 0000000..c3dd671
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/glob_subst_spec.rb
@@ -0,0 +1,12 @@
+describe 'with `GLOB_SUBST` option set' do
+ let(:after_sourcing) do
+ -> {
+ session.run_command('setopt GLOB_SUBST')
+ }
+ end
+
+ it 'error messages are not printed' do
+ session.send_string('[[')
+ wait_for { session.content }.to eq('[[')
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/rebound_bracket_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/rebound_bracket_spec.rb
new file mode 100644
index 0000000..8b420f0
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/rebound_bracket_spec.rb
@@ -0,0 +1,13 @@
+describe 'rebinding [' do
+ context 'initialized before sourcing the plugin' do
+ before do
+ session.run_command("function [ { $commands[\\[] \"$@\" }")
+ session.clear_screen
+ end
+
+ it 'executes the custom behavior and the built-in behavior' do
+ session.send_string('asdf')
+ wait_for { session.content }.to eq('asdf')
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/vi_mode_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/vi_mode_spec.rb
new file mode 100644
index 0000000..0a295c2
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/vi_mode_spec.rb
@@ -0,0 +1,80 @@
+describe 'when using vi mode' do
+ let(:before_sourcing) do
+ -> do
+ session.run_command('bindkey -v')
+ end
+ end
+
+ describe 'moving the cursor after exiting insert mode' do
+ it 'should not clear the current suggestion' do
+ with_history('foobar foo') do
+ session.
+ send_string('foo').
+ send_keys('escape').
+ send_keys('h')
+
+ wait_for { session.content }.to eq('foobar foo')
+ end
+ end
+ end
+
+ describe '`vi-forward-word-end`' do
+ it 'should accept through the end of the current word' do
+ with_history('foobar foo') do
+ session.
+ send_string('foo').
+ send_keys('escape').
+ send_keys('e'). # vi-forward-word-end
+ send_keys('a'). # vi-add-next
+ send_string('baz')
+
+ wait_for { session.content }.to eq('foobarbaz')
+ end
+ end
+ end
+
+ describe '`vi-forward-word`' do
+ it 'should accept through the first character of the next word' do
+ with_history('foobar foo') do
+ session.
+ send_string('foo').
+ send_keys('escape').
+ send_keys('w'). # vi-forward-word
+ send_keys('a'). # vi-add-next
+ send_string('az')
+
+ wait_for { session.content }.to eq('foobar faz')
+ end
+ end
+ end
+
+ describe '`vi-find-next-char`' do
+ it 'should accept through the next occurrence of the character' do
+ with_history('foobar foo') do
+ session.
+ send_string('foo').
+ send_keys('escape').
+ send_keys('f'). # vi-find-next-char
+ send_keys('o').
+ send_keys('a'). # vi-add-next
+ send_string('b')
+
+ wait_for { session.content }.to eq('foobar fob')
+ end
+ end
+ end
+
+ describe '`vi-delete`' do
+ it 'should be able to remove the last character in the buffer' do
+ skip 'deleting last char did not work below zsh version 5.0.8' if session.zsh_version < Gem::Version.new('5.0.8')
+
+ session.
+ send_string('echo foo').
+ send_keys('escape').
+ send_keys('d').
+ send_keys('l')
+
+ wait_for { session.content }.to eq('echo fo')
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/wrapped_widget_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/wrapped_widget_spec.rb
new file mode 100644
index 0000000..61dfc2d
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/wrapped_widget_spec.rb
@@ -0,0 +1,39 @@
+describe 'a wrapped widget' do
+ let(:widget) { 'backward-delete-char' }
+
+ context 'initialized before sourcing the plugin' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command("_orig_#{widget}() { zle .#{widget} }").
+ run_command("zle -N orig-#{widget} _orig_#{widget}").
+ run_command("#{widget}-magic() { zle orig-#{widget}; BUFFER+=b }").
+ run_command("zle -N #{widget} #{widget}-magic")
+ end
+ end
+
+ it 'executes the custom behavior and the built-in behavior' do
+ with_history('foobar', 'foodar') do
+ session.send_string('food').send_keys('C-h')
+ wait_for { session.content }.to eq('foobar')
+ end
+ end
+ end
+
+ context 'initialized after sourcing the plugin' do
+ before do
+ session.
+ run_command("zle -N orig-#{widget} ${widgets[#{widget}]#*:}").
+ run_command("#{widget}-magic() { zle orig-#{widget}; BUFFER+=b }").
+ run_command("zle -N #{widget} #{widget}-magic").
+ clear_screen
+ end
+
+ it 'executes the custom behavior and the built-in behavior' do
+ with_history('foobar', 'foodar') do
+ session.send_string('food').send_keys('C-h')
+ wait_for { session.content }.to eq('foobar')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/integrations/zle_input_stack_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/integrations/zle_input_stack_spec.rb
new file mode 100644
index 0000000..12cfbc7
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/integrations/zle_input_stack_spec.rb
@@ -0,0 +1,24 @@
+describe 'using `zle -U`' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('_zsh_autosuggest_strategy_test() { sleep 1; _zsh_autosuggest_strategy_history "$1" }').
+ run_command('foo() { zle -U - "echo hello" }; zle -N foo; bindkey ^B foo')
+ end
+ end
+
+ let(:options) { ['unset ZSH_AUTOSUGGEST_USE_ASYNC', 'ZSH_AUTOSUGGEST_STRATEGY=test'] }
+
+ # TODO: This is only possible with the $KEYS_QUEUED_COUNT widget parameter, coming soon...
+ xit 'does not fetch a suggestion for every inserted character' do
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('echo hello')
+ end
+
+ it 'shows a suggestion when the widget completes' do
+ with_history('echo hello world') do
+ session.send_keys('C-b')
+ wait_for { session.content(esc_seqs: true) }.to match(/\Aecho hello\e\[[0-9]+m world/)
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/kill_ring_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/kill_ring_spec.rb
new file mode 100644
index 0000000..4d0178f
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/kill_ring_spec.rb
@@ -0,0 +1,23 @@
+context 'with some items in the kill ring' do
+ before do
+ session.
+ send_string('echo foo').
+ send_keys('C-u').
+ send_string('echo bar').
+ send_keys('C-u')
+ end
+
+ describe '`yank-pop`' do
+ it 'should cycle through all items in the kill ring' do
+ session.send_keys('C-y')
+ wait_for { session.content }.to eq('echo bar')
+
+ session.send_keys('escape').send_keys('y')
+ wait_for { session.content }.to eq('echo foo')
+
+ session.send_keys('escape').send_keys('y')
+ wait_for { session.content }.to eq('echo bar')
+ end
+ end
+end
+
diff --git a/home-config/zsh/zsh-autosuggestions/spec/line_init_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/line_init_spec.rb
new file mode 100644
index 0000000..826277f
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/line_init_spec.rb
@@ -0,0 +1,17 @@
+context 'with zle-line-init unignored' do
+ let(:after_sourcing) do
+ -> do
+ session.
+ run_command('setopt extendedglob').
+ run_command('ZSH_AUTOSUGGEST_IGNORE_WIDGETS=(${(@)ZSH_AUTOSUGGEST_IGNORE_WIDGETS:#zle-\*} zle-\^line-init)').
+ run_command('zle-line-init() { BUFFER="echo" }')
+ end
+ end
+
+ it 'should fetch a suggestion on each line initialization' do
+ with_history('echo foo') do
+ session.run_command('zle -N zle-line-init')
+ wait_for { session.content }.to end_with('echo foo')
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/multi_line_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/multi_line_spec.rb
new file mode 100644
index 0000000..364780a
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/multi_line_spec.rb
@@ -0,0 +1,8 @@
+describe 'a multi-line suggestion' do
+ it 'should be displayed on multiple lines' do
+ with_history("echo \"\n\"") do
+ session.send_keys('e')
+ wait_for { session.content }.to eq("echo \"\n\"")
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/options/buffer_max_size_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/options/buffer_max_size_spec.rb
new file mode 100644
index 0000000..29ca8bc
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/options/buffer_max_size_spec.rb
@@ -0,0 +1,30 @@
+describe 'a suggestion' do
+ let(:term_opts) { { width: 200 } }
+ let(:long_command) { "echo #{'a' * 100}" }
+
+ around do |example|
+ with_history(long_command) { example.run }
+ end
+
+ it 'is provided for any buffer length' do
+ session.send_string(long_command[0...-1])
+ wait_for { session.content }.to eq(long_command)
+ end
+
+ context 'when ZSH_AUTOSUGGEST_BUFFER_MAX_SIZE is specified' do
+ let(:buffer_max_size) { 10 }
+ let(:options) { ["ZSH_AUTOSUGGEST_BUFFER_MAX_SIZE=#{buffer_max_size}"] }
+
+ it 'is provided when the buffer is shorter than the specified length' do
+ session.send_string(long_command[0...(buffer_max_size - 1)])
+ wait_for { session.content }.to eq(long_command)
+ end
+
+ it 'is provided when the buffer is equal to the specified length' do
+ session.send_string(long_command[0...(buffer_max_size)])
+ wait_for { session.content }.to eq(long_command)
+ end
+
+ it 'is not provided when the buffer is longer than the specified length'
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/options/highlight_style_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/options/highlight_style_spec.rb
new file mode 100644
index 0000000..a7e39b3
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/options/highlight_style_spec.rb
@@ -0,0 +1,7 @@
+describe 'a displayed suggestion' do
+ it 'is shown in the default style'
+
+ describe 'when ZSH_AUTOSUGGEST_HIGHLIGHT_STYLE is set to a zle_highlight string' do
+ it 'is shown in the specified style'
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/options/original_widget_prefix_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/options/original_widget_prefix_spec.rb
new file mode 100644
index 0000000..a4b6e98
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/options/original_widget_prefix_spec.rb
@@ -0,0 +1,7 @@
+describe 'an original zle widget' do
+ context 'is accessible with the default prefix'
+
+ context 'when ZSH_AUTOSUGGEST_ORIGINAL_WIDGET_PREFIX is set' do
+ it 'is accessible with the specified prefix'
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/options/strategy_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/options/strategy_spec.rb
new file mode 100644
index 0000000..58562d0
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/options/strategy_spec.rb
@@ -0,0 +1,55 @@
+describe 'a suggestion for a given prefix' do
+ let(:history_strategy) { '_zsh_autosuggest_strategy_history() { suggestion="history" }' }
+ let(:foobar_strategy) { '_zsh_autosuggest_strategy_foobar() { [[ "foobar baz" = $1* ]] && suggestion="foobar baz" }' }
+ let(:foobaz_strategy) { '_zsh_autosuggest_strategy_foobaz() { [[ "foobaz bar" = $1* ]] && suggestion="foobaz bar" }' }
+
+ let(:after_sourcing) do
+ -> do
+ session.run_command(history_strategy)
+ end
+ end
+
+ it 'by default is determined by calling the `history` strategy function' do
+ session.send_string('h')
+ wait_for { session.content }.to eq('history')
+ end
+
+ context 'when ZSH_AUTOSUGGEST_STRATEGY is set to an array' do
+ let(:after_sourcing) do
+ -> do
+ session.
+ run_command(foobar_strategy).
+ run_command(foobaz_strategy).
+ run_command('ZSH_AUTOSUGGEST_STRATEGY=(foobar foobaz)')
+ end
+ end
+
+ it 'is determined by the first strategy function to return a suggestion' do
+ session.send_string('foo')
+ wait_for { session.content }.to eq('foobar baz')
+
+ session.send_string('baz')
+ wait_for { session.content }.to eq('foobaz bar')
+ end
+ end
+
+ context 'when ZSH_AUTOSUGGEST_STRATEGY is set to a string' do
+ let(:after_sourcing) do
+ -> do
+ session.
+ run_command(foobar_strategy).
+ run_command(foobaz_strategy).
+ run_command('ZSH_AUTOSUGGEST_STRATEGY="foobar foobaz"')
+ end
+ end
+
+ it 'is determined by the first strategy function to return a suggestion' do
+ session.send_string('foo')
+ wait_for { session.content }.to eq('foobar baz')
+
+ session.send_string('baz')
+ wait_for { session.content }.to eq('foobaz bar')
+ end
+ end
+end
+
diff --git a/home-config/zsh/zsh-autosuggestions/spec/options/widget_lists_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/options/widget_lists_spec.rb
new file mode 100644
index 0000000..421b84e
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/options/widget_lists_spec.rb
@@ -0,0 +1,121 @@
+describe 'a zle widget' do
+ let(:widget) { 'my-widget' }
+ let(:before_sourcing) { -> { session.run_command("#{widget}() {}; zle -N #{widget}; bindkey ^B #{widget}") } }
+
+ context 'when added to ZSH_AUTOSUGGEST_ACCEPT_WIDGETS' do
+ let(:options) { ["ZSH_AUTOSUGGEST_ACCEPT_WIDGETS+=(#{widget})"] }
+
+ it 'accepts the suggestion and moves the cursor to the end of the buffer when invoked' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content(esc_seqs: true) }.to eq('echo hello')
+ wait_for { session.cursor }.to eq([10, 0])
+ end
+ end
+ end
+
+ context 'when added to ZSH_AUTOSUGGEST_CLEAR_WIDGETS' do
+ let(:options) { ["ZSH_AUTOSUGGEST_CLEAR_WIDGETS+=(#{widget})"] }
+
+ it 'clears the suggestion when invoked' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('e')
+ end
+ end
+ end
+
+ context 'when added to ZSH_AUTOSUGGEST_EXECUTE_WIDGETS' do
+ let(:options) { ["ZSH_AUTOSUGGEST_EXECUTE_WIDGETS+=(#{widget})"] }
+
+ it 'executes the suggestion when invoked' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content }.to end_with("\nhello")
+ end
+ end
+ end
+
+ context 'when added to ZSH_AUTOSUGGEST_IGNORE_WIDGETS' do
+ let(:options) { ["ZSH_AUTOSUGGEST_IGNORE_WIDGETS=(#{widget})"] }
+
+ it 'should not be wrapped with an autosuggest widget' do
+ session.run_command("echo $widgets[#{widget}]")
+ wait_for { session.content }.to end_with("\nuser:#{widget}")
+ end
+ end
+
+ context 'that moves the cursor forward' do
+ before { session.run_command("#{widget}() { zle forward-char }") }
+
+ context 'when added to ZSH_AUTOSUGGEST_PARTIAL_ACCEPT_WIDGETS' do
+ let(:options) { ["ZSH_AUTOSUGGEST_PARTIAL_ACCEPT_WIDGETS=(#{widget})"] }
+
+ it 'accepts the suggestion as far as the cursor is moved when invoked' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to start_with('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content(esc_seqs: true) }.to match(/\Aec\e\[[0-9]+mho hello/)
+ end
+ end
+ end
+ end
+
+ context 'that modifies the buffer' do
+ before { session.run_command("#{widget}() { BUFFER=\"foo\" }") }
+
+ context 'when not added to any of the widget lists' do
+ it 'modifies the buffer and fetches a new suggestion' do
+ with_history('foobar') do
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('foobar')
+ end
+ end
+ end
+ end
+end
+
+describe 'a modification to the widget lists' do
+ let(:widget) { 'my-widget' }
+ let(:before_sourcing) { -> { session.run_command("#{widget}() {}; zle -N #{widget}; bindkey ^B #{widget}") } }
+ before { session.run_command("ZSH_AUTOSUGGEST_ACCEPT_WIDGETS+=(#{widget})") }
+
+ it 'takes effect on the next cmd line' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content(esc_seqs: true) }.to eq('echo hello')
+ end
+ end
+
+ context 'when manual rebind is enabled' do
+ let(:options) { ["ZSH_AUTOSUGGEST_MANUAL_REBIND=true"] }
+
+ it 'does not take effect until bind command is re-run' do
+ with_history('echo hello') do
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ sleep 1
+ expect(session.content(esc_seqs: true)).not_to eq('echo hello')
+
+ session.send_keys('C-c')
+ session.run_command('_zsh_autosuggest_bind_widgets').clear_screen
+ wait_for { session.content }.to eq('')
+
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ session.send_keys('C-b')
+ wait_for { session.content(esc_seqs: true) }.to eq('echo hello')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/spec_helper.rb b/home-config/zsh/zsh-autosuggestions/spec/spec_helper.rb
new file mode 100644
index 0000000..dc1abb0
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/spec_helper.rb
@@ -0,0 +1,54 @@
+require 'pry'
+require 'rspec/wait'
+require 'terminal_session'
+require 'tempfile'
+
+RSpec.shared_context 'terminal session' do
+ let(:term_opts) { {} }
+ let(:session) { TerminalSession.new(term_opts) }
+ let(:before_sourcing) { -> {} }
+ let(:after_sourcing) { -> {} }
+ let(:options) { [] }
+
+ around do |example|
+ before_sourcing.call
+ session.run_command(['source zsh-autosuggestions.zsh', *options].join('; '))
+ after_sourcing.call
+ session.clear_screen
+
+ example.run
+
+ session.destroy
+ end
+
+ def with_history(*commands, &block)
+ Tempfile.create do |f|
+ f.write(commands.map{|c| c.gsub("\n", "\\\n")}.join("\n"))
+ f.flush
+
+ session.run_command('fc -p')
+ session.run_command("fc -R #{f.path}")
+
+ session.clear_screen
+
+ yield block
+
+ session.send_keys('C-c')
+ session.run_command('fc -P')
+ end
+ end
+end
+
+RSpec.configure do |config|
+ config.expect_with :rspec do |expectations|
+ expectations.include_chain_clauses_in_custom_matcher_descriptions = true
+ end
+
+ config.mock_with :rspec do |mocks|
+ mocks.verify_partial_doubles = true
+ end
+
+ config.wait_timeout = 2
+
+ config.include_context 'terminal session'
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/strategies/completion_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/strategies/completion_spec.rb
new file mode 100644
index 0000000..92794d6
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/strategies/completion_spec.rb
@@ -0,0 +1,72 @@
+describe 'the `completion` suggestion strategy' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=completion'] }
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('autoload compinit && compinit').
+ run_command('_foo() { compadd bar; compadd bat }').
+ run_command('_num() { compadd two; compadd three }').
+ run_command('compdef _foo baz').
+ run_command('compdef _num one')
+ end
+ end
+
+ it 'suggests the first completion result' do
+ session.send_string('baz ')
+ wait_for { session.content }.to eq('baz bar')
+ end
+
+ it 'does not add extra carriage returns when prefix has a line feed' do
+ skip '`stty` does not work inside zpty below zsh version 5.0.3' if session.zsh_version < Gem::Version.new('5.0.3')
+ session.send_string('baz \\').send_keys('C-v', 'C-j')
+ wait_for { session.content }.to eq("baz \\\nbar")
+ end
+
+ context 'when `_complete` is aliased' do
+ let(:before_sourcing) do
+ -> do
+ session.
+ run_command('autoload compinit && compinit').
+ run_command('_foo() { compadd bar; compadd bat }').
+ run_command('compdef _foo baz').
+ run_command('alias _complete=_complete')
+ end
+ end
+
+ it 'suggests the first completion result' do
+ session.send_string('baz ')
+ wait_for { session.content }.to eq('baz bar')
+ end
+ end
+
+ context 'when ZSH_AUTOSUGGEST_COMPLETION_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=completion', 'ZSH_AUTOSUGGEST_COMPLETION_IGNORE="one *"'] }
+
+ it 'makes suggestions when the buffer does not match the pattern' do
+ session.send_string('baz ')
+ wait_for { session.content }.to eq('baz bar')
+ end
+
+ it 'does not make suggestions when the buffer matches the pattern' do
+ session.send_string('one t')
+ sleep 1
+ expect(session.content).to eq('one t')
+ end
+ end
+
+ context 'when async mode is enabled' do
+ let(:options) { ['ZSH_AUTOSUGGEST_USE_ASYNC=true', 'ZSH_AUTOSUGGEST_STRATEGY=completion'] }
+
+ it 'suggests the first completion result' do
+ session.send_string('baz ')
+ wait_for { session.content }.to eq('baz bar')
+ end
+
+ it 'does not add extra carriage returns when prefix has a line feed' do
+ skip '`stty` does not work inside zpty below zsh version 5.0.3' if session.zsh_version < Gem::Version.new('5.0.3')
+ session.send_string('baz \\').send_keys('C-v', 'C-j')
+ wait_for { session.content }.to eq("baz \\\nbar")
+ end
+ end
+end
+
diff --git a/home-config/zsh/zsh-autosuggestions/spec/strategies/history_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/strategies/history_spec.rb
new file mode 100644
index 0000000..eee8efd
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/strategies/history_spec.rb
@@ -0,0 +1,23 @@
+require 'strategies/special_characters_helper'
+
+describe 'the `history` suggestion strategy' do
+ it 'suggests the last matching history entry' do
+ with_history('ls foo', 'ls bar', 'echo baz') do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls bar')
+ end
+ end
+
+ context 'when ZSH_AUTOSUGGEST_HISTORY_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_HISTORY_IGNORE="* bar"'] }
+
+ it 'does not make suggestions that match the pattern' do
+ with_history('ls foo', 'ls bar', 'echo baz') do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls foo')
+ end
+ end
+ end
+
+ include_examples 'special characters'
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/strategies/match_prev_cmd_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/strategies/match_prev_cmd_spec.rb
new file mode 100644
index 0000000..c435f16
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/strategies/match_prev_cmd_spec.rb
@@ -0,0 +1,34 @@
+require 'strategies/special_characters_helper'
+
+describe 'the `match_prev_cmd` strategy' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=match_prev_cmd'] }
+
+ let(:history) { [
+ 'echo what',
+ 'ls foo',
+ 'echo what',
+ 'ls bar',
+ 'ls baz',
+ 'echo what'
+ ] }
+
+ it 'suggests the last matching history entry after the previous command' do
+ with_history(*history) do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls bar')
+ end
+ end
+
+ context 'when ZSH_AUTOSUGGEST_HISTORY_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=match_prev_cmd', 'ZSH_AUTOSUGGEST_HISTORY_IGNORE="* bar"'] }
+
+ it 'does not make suggestions that match the pattern' do
+ with_history(*history) do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls foo')
+ end
+ end
+ end
+
+ include_examples 'special characters'
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/strategies/special_characters_helper.rb b/home-config/zsh/zsh-autosuggestions/spec/strategies/special_characters_helper.rb
new file mode 100644
index 0000000..eb1f0cd
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/strategies/special_characters_helper.rb
@@ -0,0 +1,75 @@
+shared_examples 'special characters' do
+ describe 'a special character in the buffer should be treated like any other character' do
+ it 'asterisk' do
+ with_history('echo "hello*"', 'echo "hello."') do
+ session.send_string('echo "hello*')
+ wait_for { session.content }.to eq('echo "hello*"')
+ end
+ end
+
+ it 'question mark' do
+ with_history('echo "hello?"', 'echo "hello."') do
+ session.send_string('echo "hello?')
+ wait_for { session.content }.to eq('echo "hello?"')
+ end
+ end
+
+ it 'backslash' do
+ with_history('echo "hello\nworld"') do
+ session.send_string('echo "hello\\')
+ wait_for { session.content }.to eq('echo "hello\nworld"')
+ end
+ end
+
+ it 'double backslash' do
+ with_history('echo "\\\\"') do
+ session.send_string('echo "\\\\')
+ wait_for { session.content }.to eq('echo "\\\\"')
+ end
+ end
+
+ it 'tilde' do
+ with_history('echo ~/foo') do
+ session.send_string('echo ~')
+ wait_for { session.content }.to eq('echo ~/foo')
+ end
+ end
+
+ it 'parentheses' do
+ with_history('echo "$(ls foo)"') do
+ session.send_string('echo "$(')
+ wait_for { session.content }.to eq('echo "$(ls foo)"')
+ end
+ end
+
+ it 'square bracket' do
+ with_history('echo "$history[123]"') do
+ session.send_string('echo "$history[')
+ wait_for { session.content }.to eq('echo "$history[123]"')
+ session.send_string('123]')
+ wait_for { session.content }.to eq('echo "$history[123]"')
+ end
+ end
+
+ it 'octothorpe' do
+ with_history('echo "#yolo"') do
+ session.send_string('echo "#')
+ wait_for { session.content }.to eq('echo "#yolo"')
+ end
+ end
+
+ it 'caret' do
+ with_history('echo "^A"', 'echo "^B"') do
+ session.send_string('echo "^A')
+ wait_for { session.content }.to eq('echo "^A"')
+ end
+ end
+
+ it 'dash' do
+ with_history('-foo() {}') do
+ session.send_string('-')
+ wait_for { session.content }.to eq('-foo() {}')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/terminal_session.rb b/home-config/zsh/zsh-autosuggestions/spec/terminal_session.rb
new file mode 100644
index 0000000..f91ee6c
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/terminal_session.rb
@@ -0,0 +1,99 @@
+require 'securerandom'
+
+class TerminalSession
+ ZSH_BIN = ENV['TEST_ZSH_BIN'] || 'zsh'
+
+ def initialize(opts = {})
+ opts = {
+ width: 80,
+ height: 24,
+ prompt: '',
+ term: 'xterm-256color',
+ zsh_bin: ZSH_BIN
+ }.merge(opts)
+
+ @opts = opts
+
+ cmd="PS1=\"#{opts[:prompt]}\" TERM=#{opts[:term]} #{ZSH_BIN} -f"
+ tmux_command("new-session -d -x #{opts[:width]} -y #{opts[:height]} '#{cmd}'")
+ end
+
+ def zsh_version
+ @zsh_version ||= Gem::Version.new(`#{ZSH_BIN} -c 'echo -n $ZSH_VERSION'`)
+ end
+
+ def tmux_socket_name
+ @tmux_socket_name ||= SecureRandom.hex(6)
+ end
+
+ def run_command(command)
+ send_string(command)
+ send_keys('enter')
+
+ self
+ end
+
+ def send_string(str)
+ tmux_command("send-keys -t 0 -l -- '#{str.gsub("'", "\\'")}'")
+
+ self
+ end
+
+ def send_keys(*keys)
+ tmux_command("send-keys -t 0 #{keys.join(' ')}")
+
+ self
+ end
+
+ def paste_string(str)
+ tmux_command("set-buffer -- '#{str}'")
+ tmux_command("paste-buffer -dpr -t 0")
+
+ self
+ end
+
+ def content(esc_seqs: false)
+ cmd = 'capture-pane -p -t 0'
+ cmd += ' -e' if esc_seqs
+ tmux_command(cmd).strip
+ end
+
+ def clear_screen
+ send_keys('C-l')
+
+ i = 0
+ until content == opts[:prompt] || i > 20 do
+ sleep(0.1)
+ i = i + 1
+ end
+
+ self
+ end
+
+ def destroy
+ tmux_command('kill-session')
+ end
+
+ def cursor
+ tmux_command("display-message -t 0 -p '\#{cursor_x},\#{cursor_y}'").
+ strip.
+ split(',').
+ map(&:to_i)
+ end
+
+ def attach!
+ tmux_command('attach-session')
+ end
+
+ private
+
+ attr_reader :opts
+
+ def tmux_command(cmd)
+ out = `tmux -u -L #{tmux_socket_name} #{cmd}`
+
+ raise("tmux error running: '#{cmd}'") unless $?.success?
+
+ out
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/widgets/disable_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/widgets/disable_spec.rb
new file mode 100644
index 0000000..b387a59
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/widgets/disable_spec.rb
@@ -0,0 +1,19 @@
+describe 'the `autosuggest-disable` widget' do
+ before do
+ session.run_command('bindkey ^B autosuggest-disable')
+ end
+
+ it 'disables suggestions and clears the suggestion' do
+ with_history('echo hello') do
+ session.send_string('echo')
+ wait_for { session.content }.to eq('echo hello')
+
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('echo')
+
+ session.send_string(' h')
+ sleep 1
+ expect(session.content).to eq('echo h')
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/widgets/enable_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/widgets/enable_spec.rb
new file mode 100644
index 0000000..3ad35a8
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/widgets/enable_spec.rb
@@ -0,0 +1,42 @@
+describe 'the `autosuggest-enable` widget' do
+ before do
+ session.
+ run_command('typeset -g _ZSH_AUTOSUGGEST_DISABLED').
+ run_command('bindkey ^B autosuggest-enable')
+ end
+
+ it 'enables suggestions and fetches a suggestion' do
+ with_history('echo hello') do
+ session.send_string('e')
+ sleep 1
+ expect(session.content).to eq('e')
+
+ session.send_keys('C-b')
+ session.send_string('c')
+ wait_for { session.content }.to eq('echo hello')
+ end
+ end
+
+ context 'invoked on an empty buffer' do
+ it 'does not fetch a suggestion' do
+ with_history('echo hello') do
+ session.send_keys('C-b')
+ sleep 1
+ expect(session.content).to eq('')
+ end
+ end
+ end
+
+ context 'invoked on a non-empty buffer' do
+ it 'fetches a suggestion' do
+ with_history('echo hello') do
+ session.send_string('e')
+ sleep 1
+ expect(session.content).to eq('e')
+
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('echo hello')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/widgets/fetch_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/widgets/fetch_spec.rb
new file mode 100644
index 0000000..eb8f2ba
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/widgets/fetch_spec.rb
@@ -0,0 +1,24 @@
+describe 'the `autosuggest-fetch` widget' do
+ context 'when suggestions are disabled' do
+ before do
+ session.
+ run_command('bindkey ^B autosuggest-disable').
+ run_command('bindkey ^F autosuggest-fetch').
+ send_keys('C-b')
+ end
+
+ it 'will fetch and display a suggestion' do
+ with_history('echo hello') do
+ session.send_string('echo h')
+ sleep 1
+ expect(session.content).to eq('echo h')
+
+ session.send_keys('C-f')
+ wait_for { session.content }.to eq('echo hello')
+
+ session.send_string('e')
+ wait_for { session.content }.to eq('echo hello')
+ end
+ end
+ end
+end
diff --git a/home-config/zsh/zsh-autosuggestions/spec/widgets/toggle_spec.rb b/home-config/zsh/zsh-autosuggestions/spec/widgets/toggle_spec.rb
new file mode 100644
index 0000000..8f9f3c3
--- /dev/null
+++ b/home-config/zsh/zsh-autosuggestions/spec/widgets/toggle_spec.rb
@@ -0,0 +1,26 @@
+describe 'the `autosuggest-toggle` widget' do
+ before do
+ session.run_command('bindkey ^B autosuggest-toggle')
+ end
+
+ it 'toggles suggestions' do
+ with_history('echo world', 'echo hello') do
+ session.send_string('echo')
+ wait_for { session.content }.to eq('echo hello')
+
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('echo')
+
+ session.send_string(' h')
+ sleep 1
+ expect(session.content).to eq('echo h')
+
+ session.send_keys('C-b')
+ wait_for { session.content }.to eq('echo hello')
+
+ session.send_keys('C-h')
+ session.send_string('w')
+ wait_for { session.content }.to eq('echo world')
+ end
+ end
+end