Merge pull request #420 from zsh-users/fixes/completion-newline-issues

Fixes/completion newline issues
This commit is contained in:
Eric Freese 2019-06-15 21:24:01 -06:00 committed by GitHub
commit 86f64c30ca
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 75 additions and 15 deletions

View file

@ -4,7 +4,7 @@ describe 'the `completion` suggestion strategy' do
-> do -> do
session. session.
run_command('autoload compinit && compinit'). run_command('autoload compinit && compinit').
run_command('_foo() { compadd bar }'). run_command('_foo() { compadd bar; compadd bat }').
run_command('compdef _foo baz') run_command('compdef _foo baz')
end end
end end
@ -14,6 +14,12 @@ describe 'the `completion` suggestion strategy' do
wait_for { session.content }.to eq('baz bar') wait_for { session.content }.to eq('baz bar')
end 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 async mode is enabled' do context 'when async mode is enabled' do
let(:options) { ['ZSH_AUTOSUGGEST_USE_ASYNC=true', 'ZSH_AUTOSUGGEST_STRATEGY=completion'] } let(:options) { ['ZSH_AUTOSUGGEST_USE_ASYNC=true', 'ZSH_AUTOSUGGEST_STRATEGY=completion'] }
@ -21,6 +27,12 @@ describe 'the `completion` suggestion strategy' do
session.send_string('baz ') session.send_string('baz ')
wait_for { session.content }.to eq('baz bar') wait_for { session.content }.to eq('baz bar')
end 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
end end

View file

@ -56,9 +56,12 @@ _zsh_autosuggest_async_request() {
_zsh_autosuggest_async_response() { _zsh_autosuggest_async_response() {
emulate -L zsh emulate -L zsh
local suggestion
if [[ -z "$2" || "$2" == "hup" ]]; then if [[ -z "$2" || "$2" == "hup" ]]; then
# Read everything from the fd and give it as a suggestion # Read everything from the fd and give it as a suggestion
zle autosuggest-suggest -- "$(cat <&$1)" IFS='' read -rd '' -u $1 suggestion
zle autosuggest-suggest -- "$suggestion"
# Close the fd # Close the fd
exec {1}<&- exec {1}<&-

View file

@ -10,10 +10,13 @@ _zsh_autosuggest_capture_postcompletion() {
compstate[insert]=1 compstate[insert]=1
# Don't list completions # Don't list completions
unset compstate[list] unset 'compstate[list]'
} }
_zsh_autosuggest_capture_completion_widget() { _zsh_autosuggest_capture_completion_widget() {
# Add a post-completion hook to be called after all completions have been
# gathered. The hook can modify compstate to affect what is done with the
# gathered completions.
local -a +h comppostfuncs local -a +h comppostfuncs
comppostfuncs=(_zsh_autosuggest_capture_postcompletion) comppostfuncs=(_zsh_autosuggest_capture_postcompletion)
@ -25,6 +28,16 @@ _zsh_autosuggest_capture_completion_widget() {
# after autosuggestions is initialized. # after autosuggestions is initialized.
zle -- ${(k)widgets[(r)completion:.complete-word:_main_complete]} zle -- ${(k)widgets[(r)completion:.complete-word:_main_complete]}
if is-at-least 5.0.3; then
# Don't do any cr/lf transformations. We need to do this immediately before
# output because if we do it in setup, onlcr will be re-enabled when we enter
# vared in the async code path. There is a bug in zpty module in older versions
# where the tty is not properly attached to the pty slave, resulting in stty
# getting stopped with a SIGTTOU. See zsh-workers thread 31660 and upstream
# commit f75904a38
stty -onlcr -ocrnl -F /dev/tty
fi
# The completion has been added, print the buffer as the suggestion # The completion has been added, print the buffer as the suggestion
echo -nE - $'\0'$BUFFER$'\0' echo -nE - $'\0'$BUFFER$'\0'
} }
@ -32,6 +45,8 @@ _zsh_autosuggest_capture_completion_widget() {
zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget
_zsh_autosuggest_capture_setup() { _zsh_autosuggest_capture_setup() {
autoload -Uz is-at-least
# There is a bug in zpty module in older zsh versions by which a # There is a bug in zpty module in older zsh versions by which a
# zpty that exits will kill all zpty processes that were forked # zpty that exits will kill all zpty processes that were forked
# before it. Here we set up a zsh exit hook to SIGKILL the zpty # before it. Here we set up a zsh exit hook to SIGKILL the zpty
@ -39,8 +54,12 @@ _zsh_autosuggest_capture_setup() {
# zpty processes. # zpty processes.
if ! is-at-least 5.4; then if ! is-at-least 5.4; then
zshexit() { zshexit() {
kill -KILL $$ # The zsh builtin `kill` fails sometimes in older versions
sleep 1 # Block for long enough for the signal to come through # https://unix.stackexchange.com/a/477647/156673
kill -KILL $$ 2>&- || command kill -KILL $$
# Block for long enough for the signal to come through
sleep 1
} }
fi fi
@ -99,9 +118,11 @@ _zsh_autosuggest_strategy_completion() {
# content between the first two null bytes. # content between the first two null bytes.
zpty -r $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME line '*'$'\0''*'$'\0' zpty -r $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME line '*'$'\0''*'$'\0'
# On older versions of zsh, we sometimes get extra bytes after the # Extract the suggestion from between the null bytes. On older
# second null byte, so trim those off the end # versions of zsh (older than 5.3), we sometimes get extra bytes after
suggestion="${${${(M)line:#*$'\0'*$'\0'*}#*$'\0'}%%$'\0'*}" # the second null byte, so trim those off the end.
# See http://www.zsh.org/mla/workers/2015/msg03290.html
suggestion="${${line#*$'\0'}%$'\0'*}"
} always { } always {
# Destroy the pty # Destroy the pty
zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME

View file

@ -492,10 +492,13 @@ _zsh_autosuggest_capture_postcompletion() {
compstate[insert]=1 compstate[insert]=1
# Don't list completions # Don't list completions
unset compstate[list] unset 'compstate[list]'
} }
_zsh_autosuggest_capture_completion_widget() { _zsh_autosuggest_capture_completion_widget() {
# Add a post-completion hook to be called after all completions have been
# gathered. The hook can modify compstate to affect what is done with the
# gathered completions.
local -a +h comppostfuncs local -a +h comppostfuncs
comppostfuncs=(_zsh_autosuggest_capture_postcompletion) comppostfuncs=(_zsh_autosuggest_capture_postcompletion)
@ -507,6 +510,16 @@ _zsh_autosuggest_capture_completion_widget() {
# after autosuggestions is initialized. # after autosuggestions is initialized.
zle -- ${(k)widgets[(r)completion:.complete-word:_main_complete]} zle -- ${(k)widgets[(r)completion:.complete-word:_main_complete]}
if is-at-least 5.0.3; then
# Don't do any cr/lf transformations. We need to do this immediately before
# output because if we do it in setup, onlcr will be re-enabled when we enter
# vared in the async code path. There is a bug in zpty module in older versions
# where the tty is not properly attached to the pty slave, resulting in stty
# getting stopped with a SIGTTOU. See zsh-workers thread 31660 and upstream
# commit f75904a38
stty -onlcr -ocrnl -F /dev/tty
fi
# The completion has been added, print the buffer as the suggestion # The completion has been added, print the buffer as the suggestion
echo -nE - $'\0'$BUFFER$'\0' echo -nE - $'\0'$BUFFER$'\0'
} }
@ -514,6 +527,8 @@ _zsh_autosuggest_capture_completion_widget() {
zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget
_zsh_autosuggest_capture_setup() { _zsh_autosuggest_capture_setup() {
autoload -Uz is-at-least
# There is a bug in zpty module in older zsh versions by which a # There is a bug in zpty module in older zsh versions by which a
# zpty that exits will kill all zpty processes that were forked # zpty that exits will kill all zpty processes that were forked
# before it. Here we set up a zsh exit hook to SIGKILL the zpty # before it. Here we set up a zsh exit hook to SIGKILL the zpty
@ -521,8 +536,12 @@ _zsh_autosuggest_capture_setup() {
# zpty processes. # zpty processes.
if ! is-at-least 5.4; then if ! is-at-least 5.4; then
zshexit() { zshexit() {
kill -KILL $$ # The zsh builtin `kill` fails sometimes in older versions
sleep 1 # Block for long enough for the signal to come through # https://unix.stackexchange.com/a/477647/156673
kill -KILL $$ 2>&- || command kill -KILL $$
# Block for long enough for the signal to come through
sleep 1
} }
fi fi
@ -581,9 +600,11 @@ _zsh_autosuggest_strategy_completion() {
# content between the first two null bytes. # content between the first two null bytes.
zpty -r $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME line '*'$'\0''*'$'\0' zpty -r $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME line '*'$'\0''*'$'\0'
# On older versions of zsh, we sometimes get extra bytes after the # Extract the suggestion from between the null bytes. On older
# second null byte, so trim those off the end # versions of zsh (older than 5.3), we sometimes get extra bytes after
suggestion="${${${(M)line:#*$'\0'*$'\0'*}#*$'\0'}%%$'\0'*}" # the second null byte, so trim those off the end.
# See http://www.zsh.org/mla/workers/2015/msg03290.html
suggestion="${${line#*$'\0'}%$'\0'*}"
} always { } always {
# Destroy the pty # Destroy the pty
zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME
@ -758,9 +779,12 @@ _zsh_autosuggest_async_request() {
_zsh_autosuggest_async_response() { _zsh_autosuggest_async_response() {
emulate -L zsh emulate -L zsh
local suggestion
if [[ -z "$2" || "$2" == "hup" ]]; then if [[ -z "$2" || "$2" == "hup" ]]; then
# Read everything from the fd and give it as a suggestion # Read everything from the fd and give it as a suggestion
zle autosuggest-suggest -- "$(cat <&$1)" IFS='' read -rd '' -u $1 suggestion
zle autosuggest-suggest -- "$suggestion"
# Close the fd # Close the fd
exec {1}<&- exec {1}<&-