From 432b0755d33f5d7c0f9e085939338aa0644c17e9 Mon Sep 17 00:00:00 2001 From: musjj <72612857+musjj@users.noreply.github.com> Date: Wed, 19 Apr 2023 08:20:41 +0700 Subject: [PATCH 1/2] feat(preview-tui): handle quoting in start_preview more robustly This commit makes the script more resistant to naughty filenames. The script now depends on bash for the following features: - Arrays Correctly creating and passing argument lists is now simple - Parameter transformation `${parameter@Q}` makes it easy to correctly quote a string so that it can be safely re-evaluated by the interpreter later. On iTerm, the shell command used to render the preview is now passed to osascript via a named pipe: `$FIFO_OSASCRIPT`. By not embedding the shell command directly, we now no longer need to worry about osascript's quoting rules. It's not perfect, because $SHELL and $TMPDIR might contain naughty characters, but it's quite unlikely to happen. --- plugins/preview-tui | 86 +++++++++++++++++++++++---------------------- 1 file changed, 44 insertions(+), 42 deletions(-) diff --git a/plugins/preview-tui b/plugins/preview-tui index 6a4be8b5..3a43e2f2 100755 --- a/plugins/preview-tui +++ b/plugins/preview-tui @@ -1,4 +1,4 @@ -#!/usr/bin/env sh +#!/usr/bin/env bash # Description: Terminal based file previewer # @@ -89,27 +89,29 @@ NNN_SPLITSIZE=${NNN_SPLITSIZE:-50} # Set previewer split size percentage TMPDIR=${TMPDIR:-/tmp} NNN_PARENT=${NNN_FIFO#*.} [ "$NNN_PARENT" -eq "$NNN_PARENT" ] 2>/dev/null || NNN_PARENT="" # Make empty if non-numeric -ENVVARS=" -PWD=$PWD -PATH=$PATH -PREVIEW_MODE=$2 -NNN_FIFO=$NNN_FIFO -NNN_SCOPE=${NNN_SCOPE:-0} -NNN_PISTOL=${NNN_PISTOL:-0} -NNN_ICONLOOKUP=${NNN_ICONLOOKUP:-0} -NNN_PAGER=${NNN_PAGER:-less -P?n -R} -NNN_BATTHEME=${NNN_BATTHEME:-ansi} -NNN_BATSTYLE=${NNN_BATSTYLE:-numbers} -NNN_PREVIEWWIDTH=${NNN_PREVIEWWIDTH:-1920} -NNN_PREVIEWHEIGHT=${NNN_PREVIEWHEIGHT:-1080} -NNN_PREVIEWDIR=${NNN_PREVIEWDIR:-$TMPDIR/nnn/previews} -NNN_PREVIEWIMGPROG=${NNN_PREVIEWIMGPROG:-} -FIFOPID=$TMPDIR/nnn-preview-tui-fifopid.$NNN_PARENT -FIFOPATH=$TMPDIR/nnn-preview-tui-fifo.$NNN_PARENT -PREVIEWPID=$TMPDIR/nnn-preview-tui-previewpid.$NNN_PARENT -CURSEL=$TMPDIR/nnn-preview-tui-selection.$NNN_PARENT -FIFO_UEBERZUG=$TMPDIR/nnn-preview-tui-ueberzug-fifo.$NNN_PARENT -POSOFFSET=$TMPDIR/nnn-preview-tui-posoffset" +ENVVARS=( + "PWD=$PWD" + "PATH=$PATH" + "PREVIEW_MODE=$2" + "NNN_FIFO=$NNN_FIFO" + "NNN_SCOPE=${NNN_SCOPE:-0}" + "NNN_PISTOL=${NNN_PISTOL:-0}" + "NNN_ICONLOOKUP=${NNN_ICONLOOKUP:-0}" + "NNN_PAGER=${NNN_PAGER:-less -P?n -R}" + "NNN_BATTHEME=${NNN_BATTHEME:-ansi}" + "NNN_BATSTYLE=${NNN_BATSTYLE:-numbers}" + "NNN_PREVIEWWIDTH=${NNN_PREVIEWWIDTH:-1920}" + "NNN_PREVIEWHEIGHT=${NNN_PREVIEWHEIGHT:-1080}" + "NNN_PREVIEWDIR=${NNN_PREVIEWDIR:-$TMPDIR/nnn/previews}" + "NNN_PREVIEWIMGPROG=${NNN_PREVIEWIMGPROG:-}" + "FIFOPID=$TMPDIR/nnn-preview-tui-fifopid.$NNN_PARENT" + "FIFOPATH=$TMPDIR/nnn-preview-tui-fifo.$NNN_PARENT" + "PREVIEWPID=$TMPDIR/nnn-preview-tui-previewpid.$NNN_PARENT" + "CURSEL=$TMPDIR/nnn-preview-tui-selection.$NNN_PARENT" + "FIFO_UEBERZUG=$TMPDIR/nnn-preview-tui-ueberzug-fifo.$NNN_PARENT" + "FIFO_OSASCRIPT=$TMPDIR/nnn-preview-tui-osascript-fifo.$NNN_PARENT" + "POSOFFSET=$TMPDIR/nnn-preview-tui-posoffset" +) if [ -e "${TMUX%%,*}" ] && tmux -V | grep -q '[ -][3456789]\.'; then NNN_TERMINAL=tmux @@ -119,6 +121,7 @@ elif [ -n "$WEZTERM_PANE" ]; then NNN_TERMINAL=wezterm elif [ -z "$NNN_TERMINAL" ] && [ "$TERM_PROGRAM" = "iTerm.app" ]; then NNN_TERMINAL=iterm + mkfifo "$FIFO_OSASCRIPT" || exit 1 elif [ -n "$WT_SESSION" ]; then NNN_TERMINAL=winterm else @@ -131,20 +134,19 @@ elif [ "$NNN_SPLIT" != 'h' ]; then NNN_SPLIT='v' fi -ENVVARS="$ENVVARS -NNN_SPLIT=$NNN_SPLIT -NNN_TERMINAL=$NNN_TERMINAL" -IFS=' -' -for env in $ENVVARS; do +ENVVARS+=( + "NNN_SPLIT=$NNN_SPLIT" + "NNN_TERMINAL=$NNN_TERMINAL" +) +ENVARGS=() +for env in "${ENVVARS[@]}"; do export "${env?}" case "$NNN_TERMINAL" in - tmux) ENVSTRING="$ENVSTRING -e '$env'" ;; - kitty) ENVSTRING="$ENVSTRING --env '$env'" ;; - winterm|iterm) ENVSTRING="$ENVSTRING \\\"$env\\\"" ;; - *) ENVSTRING="$ENVSTRING $env";; + tmux) ENVARGS+=(-e "$env") ;; + kitty) ENVARGS+=(--env "$env") ;; + winterm|iterm|*) ENVARGS+=("$env") ;; esac -done; unset IFS +done trap '' PIPE exists() { type "$1" >/dev/null 2>&1 ;} @@ -165,35 +167,35 @@ start_preview() { case "$NNN_TERMINAL" in tmux) # tmux splits are inverted if [ "$NNN_SPLIT" = "v" ]; then split="h"; else split="v"; fi - eval tmux split-window "$ENVSTRING" -d"$split" -p"$NNN_SPLITSIZE" "$0" "$1" 1 ;; + tmux split-window "${ENVARGS[@]}" -d"$split" -p"$NNN_SPLITSIZE" "$0" "$1" 1 ;; kitty) # Setting the layout for the new window. It will be restored after the script ends. kitty @ goto-layout splits # Trying to use kitty's integrated window management as the split window. - eval kitty @ launch --no-response --title "preview-tui" --keep-focus \ - --cwd "$PWD" "$ENVSTRING" --location "${NNN_SPLIT}split" "$0" "$1" 1 ;; + kitty @ launch --no-response --title "preview-tui" --keep-focus \ + --cwd "$PWD" "${ENVARGS[@]}" --location "${NNN_SPLIT}split" "$0" "$1" 1 ;; wezterm) if [ "$NNN_SPLIT" = "v" ]; then split="--horizontal"; else split="--bottom"; fi wezterm cli split-pane --cwd "$PWD" $split --percent "$NNN_SPLITSIZE" "$0" "$1" 1 >/dev/null wezterm cli activate-pane-direction Prev ;; iterm) - command="$SHELL -c 'cd $PWD; env $ENVSTRING $0 $1 1'" + echo "cd ${PWD@Q}; env ${ENVARGS[*]@Q} ${0@Q} ${1@Q} 1" > "$FIFO_OSASCRIPT" & if [ "$NNN_SPLIT" = "h" ]; then split="horizontally"; else split="vertically"; fi osascript <<-EOF tell application "iTerm" tell current session of current window - split $split with default profile command "$command" + split $split with default profile command "$SHELL $FIFO_OSASCRIPT" end tell end tell EOF ;; winterm) if [ "$NNN_SPLIT" = "h" ]; then split="H"; else split="V"; fi - cmd.exe /c wt -w 0 sp -$split -s"0.$NNN_SPLITSIZE" bash -c "cd $PWD \; \ - env $ENVSTRING QLPATH=$2 $0 $1 1" \; -w 0 mf previous 2>/dev/null ;; + wt -w 0 sp -$split -s"0.$NNN_SPLITSIZE" bash -c "cd ${PWD@Q} ; \ + env ${ENVARGS[*]@Q} QLPATH=${2@Q} ${0@Q} ${1@Q} 1" \; -w 0 mf previous 2>/dev/null ;; *) if [ -n "$2" ]; then - env "$ENVSTRING" QUICKLOOK=1 QLPATH="$2" "$0" "$1" 1 & + env "${ENVARGS[@]}" QUICKLOOK=1 QLPATH="$2" "$0" "$1" 1 & else - env "$ENVSTRING" "$NNN_TERMINAL" -e "$0" "$1" 1 & + env "${ENVARGS[@]}" "$NNN_TERMINAL" -e "$0" "$1" 1 & fi ;; esac } From d3b5d0e49da457c3815e581226dd14717bf1b36a Mon Sep 17 00:00:00 2001 From: Luuk van Baal Date: Thu, 20 Apr 2023 01:06:43 +0200 Subject: [PATCH 2/2] perf(preview-tui): replace env for loop with parameter expansion --- plugins/preview-tui | 120 ++++++++++++++++++++------------------------ 1 file changed, 54 insertions(+), 66 deletions(-) diff --git a/plugins/preview-tui b/plugins/preview-tui index 3a43e2f2..a8b68890 100755 --- a/plugins/preview-tui +++ b/plugins/preview-tui @@ -13,7 +13,7 @@ # - Windows Terminal (https://github.com/Microsoft/Terminal | https://aka.ms/terminal) with WSL, or # - $NNN_TERMINAL set to a terminal (it's xterm by default). # - less or $NNN_PAGER -# - tree or exa or ls +# - tree or exa or (GNU) ls # - mediainfo or file # - mktemp # - unzip @@ -61,10 +61,6 @@ # will try to use a kitty terminal split. And as a final fallback, a # different terminal window will be used ($NNN_TERMINAL). # -# Tmux, wezterm and kitty users can configure $NNN_SPLIT to either "h" or "v" to set a -# 'h'orizontal split or a 'v'ertical split (as in, the line that splits the -# windows will be horizontal or vertical). -# # Kitty users need something similar to the following in their kitty.conf: # - `allow_remote_control yes` # - `listen_on unix:$TMPDIR/kitty` @@ -75,12 +71,14 @@ # Wezterm should work out of the box. If `NNN_PREVIEWIMGPROG` is not specified it will use # built in iTerm2 image protocol. # -# Iterm2 users are recommended to use viu to view images without getting pixelated. +# Note that GNU ls is used for its `--group-directories-first` flag. +# On MacOS this may be installed with `brew install coreutils`, or the flag can be removed. +# iTerm2 users are recommended to use viu to view images without getting pixelated. # # Windows Terminal users can set "Profile termination behavior" under "Profile > Advanced" settings # to automatically close pane on quit when exit code is 0. # -# Shell: POSIX compliant +# Shell: Bash (for environment manipulation through arrays) # Authors: Todd Yamakawa, Léo Villeveygoux, @Recidiviste, Mario Ortiz Manero, Luuk van Baal, @WanderLanz NNN_SPLIT=${NNN_SPLIT:-} # Set permanent split direction @@ -92,7 +90,6 @@ NNN_PARENT=${NNN_FIFO#*.} ENVVARS=( "PWD=$PWD" "PATH=$PATH" - "PREVIEW_MODE=$2" "NNN_FIFO=$NNN_FIFO" "NNN_SCOPE=${NNN_SCOPE:-0}" "NNN_PISTOL=${NNN_PISTOL:-0}" @@ -109,98 +106,92 @@ ENVVARS=( "PREVIEWPID=$TMPDIR/nnn-preview-tui-previewpid.$NNN_PARENT" "CURSEL=$TMPDIR/nnn-preview-tui-selection.$NNN_PARENT" "FIFO_UEBERZUG=$TMPDIR/nnn-preview-tui-ueberzug-fifo.$NNN_PARENT" - "FIFO_OSASCRIPT=$TMPDIR/nnn-preview-tui-osascript-fifo.$NNN_PARENT" "POSOFFSET=$TMPDIR/nnn-preview-tui-posoffset" ) -if [ -e "${TMUX%%,*}" ] && tmux -V | grep -q '[ -][3456789]\.'; then - NNN_TERMINAL=tmux -elif [ -n "$KITTY_LISTEN_ON" ]; then - NNN_TERMINAL=kitty -elif [ -n "$WEZTERM_PANE" ]; then - NNN_TERMINAL=wezterm -elif [ -z "$NNN_TERMINAL" ] && [ "$TERM_PROGRAM" = "iTerm.app" ]; then - NNN_TERMINAL=iterm - mkfifo "$FIFO_OSASCRIPT" || exit 1 -elif [ -n "$WT_SESSION" ]; then - NNN_TERMINAL=winterm -else - NNN_TERMINAL="${NNN_TERMINAL:-xterm}" -fi - -if [ -z "$NNN_SPLIT" ] && [ $(($(tput lines) * 2)) -gt "$(tput cols)" ]; then - NNN_SPLIT='h' -elif [ "$NNN_SPLIT" != 'h' ]; then - NNN_SPLIT='v' -fi - -ENVVARS+=( - "NNN_SPLIT=$NNN_SPLIT" - "NNN_TERMINAL=$NNN_TERMINAL" -) -ENVARGS=() -for env in "${ENVVARS[@]}"; do - export "${env?}" - case "$NNN_TERMINAL" in - tmux) ENVARGS+=(-e "$env") ;; - kitty) ENVARGS+=(--env "$env") ;; - winterm|iterm|*) ENVARGS+=("$env") ;; - esac -done - trap '' PIPE exists() { type "$1" >/dev/null 2>&1 ;} pkill() { command pkill "$@" >/dev/null 2>&1 ;} -prompt() { printf "%b" "$@"; cfg=$(stty -g); stty raw -echo; head -c 1; stty "$cfg" ;} +prompt() { clear; printf "%b" "$@"; cfg=$(stty -g); stty raw -echo; head -c 1; stty "$cfg" ;} pidkill() { - if [ -f "$1" ]; then - PID="$(cat "$1" 2>/dev/null)" || return 1 - kill "$PID" >/dev/null 2>&1 - RET=$? - wait "$PID" 2>/dev/null - return $RET - fi - return 1 + if [ -f "$1" ]; then + PID="$(cat "$1" 2>/dev/null)" || return 1 + kill "$PID" >/dev/null 2>&1 + RET=$? + wait "$PID" 2>/dev/null + return $RET + fi + return 1 } start_preview() { + if [ -e "${TMUX%%,*}" ] && tmux -V | grep -q '[ -][3456789]\.'; then + NNN_TERMINAL=tmux + elif [ -n "$KITTY_LISTEN_ON" ]; then + NNN_TERMINAL=kitty + elif [ -n "$WEZTERM_PANE" ]; then + NNN_TERMINAL=wezterm + elif [ -z "$NNN_TERMINAL" ] && [ "$TERM_PROGRAM" = "iTerm.app" ]; then + NNN_TERMINAL=iterm + elif [ -n "$WT_SESSION" ]; then + NNN_TERMINAL=winterm + else + NNN_TERMINAL="${NNN_TERMINAL:-xterm}" + fi + + if [ -z "$NNN_SPLIT" ] && [ $(($(tput lines) * 2)) -gt "$(tput cols)" ]; then + NNN_SPLIT='h' + elif [ "$NNN_SPLIT" != 'h' ]; then + NNN_SPLIT='v' + fi + + ENVVARS+=("NNN_TERMINAL=$NNN_TERMINAL" "NNN_SPLIT=$NNN_SPLIT" "QLPATH=$2" "PREVIEW_MODE=1") + case "$NNN_TERMINAL" in + iterm|winterm) # need run in separate shell command: escape + ENVVARS=("${ENVVARS[@]/#/\\\"}") + ENVVARS=("${ENVVARS[@]/%/\\\"}") + command="$SHELL -c 'env ${ENVVARS[*]} \\\"$0\\\" \\\"$1\\\"'" ;; + esac + case "$NNN_TERMINAL" in tmux) # tmux splits are inverted + ENVVARS=("${ENVVARS[@]/#/-e}") if [ "$NNN_SPLIT" = "v" ]; then split="h"; else split="v"; fi - tmux split-window "${ENVARGS[@]}" -d"$split" -p"$NNN_SPLITSIZE" "$0" "$1" 1 ;; + tmux split-window "${ENVVARS[@]}" -d"$split" -p"$NNN_SPLITSIZE" "$0" "$1" ;; kitty) # Setting the layout for the new window. It will be restored after the script ends. + ENVVARS=("${ENVVARS[@]/#/--env=}") kitty @ goto-layout splits # Trying to use kitty's integrated window management as the split window. kitty @ launch --no-response --title "preview-tui" --keep-focus \ - --cwd "$PWD" "${ENVARGS[@]}" --location "${NNN_SPLIT}split" "$0" "$1" 1 ;; + --cwd "$PWD" "${ENVVARS[@]}" --location "${NNN_SPLIT}split" "$0" "$1" ;; wezterm) + export "${ENVVARS[@]}" if [ "$NNN_SPLIT" = "v" ]; then split="--horizontal"; else split="--bottom"; fi - wezterm cli split-pane --cwd "$PWD" $split --percent "$NNN_SPLITSIZE" "$0" "$1" 1 >/dev/null + wezterm cli split-pane --cwd "$PWD" $split --percent "$NNN_SPLITSIZE" "$0" "$1" >/dev/null wezterm cli activate-pane-direction Prev ;; iterm) - echo "cd ${PWD@Q}; env ${ENVARGS[*]@Q} ${0@Q} ${1@Q} 1" > "$FIFO_OSASCRIPT" & if [ "$NNN_SPLIT" = "h" ]; then split="horizontally"; else split="vertically"; fi osascript <<-EOF tell application "iTerm" tell current session of current window - split $split with default profile command "$SHELL $FIFO_OSASCRIPT" + split $split with default profile command "$command" end tell end tell EOF ;; winterm) if [ "$NNN_SPLIT" = "h" ]; then split="H"; else split="V"; fi - wt -w 0 sp -$split -s"0.$NNN_SPLITSIZE" bash -c "cd ${PWD@Q} ; \ - env ${ENVARGS[*]@Q} QLPATH=${2@Q} ${0@Q} ${1@Q} 1" \; -w 0 mf previous 2>/dev/null ;; + wt -w 0 sp -$split -s"0.$NNN_SPLITSIZE" "$command" \; -w 0 mf previous 2>/dev/null ;; *) if [ -n "$2" ]; then - env "${ENVARGS[@]}" QUICKLOOK=1 QLPATH="$2" "$0" "$1" 1 & + env "${ENVVARS[@]}" QUICKLOOK=1 "$0" "$1" & else - env "${ENVARGS[@]}" "$NNN_TERMINAL" -e "$0" "$1" 1 & + env "${ENVVARS[@]}" "$NNN_TERMINAL" -e "$0" "$1" & fi ;; esac } toggle_preview() { + export "${ENVVARS[@]}" if exists QuickLook.exe; then QLPATH="QuickLook.exe" elif exists Bridge.exe; then @@ -417,8 +408,7 @@ generate_preview() { image_preview() { clear exec >/dev/tty - if [ "$NNN_TERMINAL" = "kitty" ]; then - # Kitty terminal users can use the native image preview method + if [ "$NNN_TERMINAL" = "kitty" ] && [ -z "$NNN_PREVIEWIMGPROG" ]; then kitty +kitten icat --silent --scale-up --place "$1"x"$2"@0x0 --transfer-mode=stream --stdin=no "$3" & elif [ "$NNN_TERMINAL" = "wezterm" ] && [ -z "$NNN_PREVIEWIMGPROG" ]; then wezterm imgcat "$3" & @@ -486,10 +476,8 @@ if [ "$PREVIEW_MODE" -eq 1 ] 2>/dev/null; then exit 0 else if [ ! -r "$NNN_FIFO" ]; then - clear prompt "No FIFO available! (\$NNN_FIFO='$NNN_FIFO')\nPlease read Usage in '$0'." elif [ "$KITTY_WINDOW_ID" ] && [ -z "$TMUX" ] && [ -z "$KITTY_LISTEN_ON" ]; then - clear prompt "\$KITTY_LISTEN_ON not set!\nPlease read Usage in '$0'." else toggle_preview "$1" &