Подтвердить что ты не робот

Умнее восстановление Vim?

Когда предыдущий сеанс Vim разбился, вас встречает "файл свопинга... уже существует!". для каждого файла, который был открыт в предыдущем сеансе.

Можете ли вы сделать этот запрос восстановления Vim умнее? (Без отключения восстановления!) В частности, я думаю:

  • Если версия с заменой не содержит несохраненных изменений и процесс редактирования больше не работает, можете ли вы сделать Vim автоматически удалять файл подкачки?
  • Можете ли вы автоматизировать предлагаемый процесс сохранения восстановленного файла под новым именем, слияния его с файлом на диске и последующего удаления старого файла подкачки, чтобы требовалось минимальное взаимодействие? Особенно, когда версия подкачки и версия диска одинаковы, все должно быть автоматическим.

Я обнаружил автокоманду SwapExists, но я не знаю, поможет ли она этим задачам.

4b9b3361

Ответ 1

У меня есть vim, хранящий мои файлы подкачки в одном локальном каталоге, имея это в моем .vimrc:

set directory=~/.vim/swap,.

Среди других преимуществ это упрощает поиск файлов подкачки сразу. Теперь, когда мой ноутбук теряет власть или что-то еще, и я запускаю резервную копию с кучей файлов подкачки, я просто запускаю свой cleanswap script:

TMPDIR=$(mktemp -d) || exit 1
RECTXT="$TMPDIR/vim.recovery.$USER.txt"
RECFN="$TMPDIR/vim.recovery.$USER.fn"
trap 'rm -f "$RECTXT" "$RECFN"; rmdir "$TMPDIR"' 0 1 2 3 15
for q in ~/.vim/swap/.*sw? ~/.vim/swap/*; do
  [[ -f $q ]] || continue
  rm -f "$RECTXT" "$RECFN"
  vim -X -r "$q" \
      -c "w! $RECTXT" \
      -c "let fn=expand('%')" \
      -c "new $RECFN" \
      -c "exec setline( 1, fn )" \
      -c w\! \
      -c "qa"
  if [[ ! -f $RECFN ]]; then
    echo "nothing to recover from $q"
    rm -f "$q"
    continue
  fi
  CRNT="$(cat $RECFN)"
  if diff --strip-trailing-cr --brief "$CRNT" "$RECTXT"; then
      echo "removing redundant $q"
      echo "  for $CRNT"
      rm -f "$q"
  else
      echo $q contains changes
      vim -n -d "$CRNT" "$RECTXT"
      rm -i "$q" || exit
  fi
done

Это приведет к удалению любых файлов подкачки, актуальных для реальных файлов. Любые, которые не совпадают, отображаются в окне vimdiff, поэтому я могу объединить свои несохраненные изменения.

- Chouser

Ответ 2

Я только что обнаружил это:

http://vimdoc.sourceforge.net/htmldoc/diff.html#:DiffOrig

Я скопировал и вставил команду DiffOrig в мой .vimrc файл, и он работает как шарм. Это значительно облегчает восстановление файлов подкачки. Я не знаю, почему он не включен по умолчанию в VIM.

Здесь команда для тех, кто спешит:

 command DiffOrig vert new | set bt=nofile | r # | 0d_ | diffthis
    \ | wincmd p | diffthis

Ответ 3

Принятый ответ выдается за очень важный прецедент. Скажем, вы создаете новый буфер и набираете 2 часа без сохранения, тогда ваш ноутбук падает. Если вы запустите предлагаемый script , он удалит вашу единственную запись, файл подкачки .swp. Я не уверен, что такое правильное исправление, но похоже, что команда diff в конечном итоге сравнивает один и тот же файл с самим собой. Измененная версия ниже проверяет этот случай и дает пользователю возможность сохранить файл где-то.

#!/bin/bash

SWAP_FILE_DIR=~/temp/vim_swp
IFS=$'\n'

TMPDIR=$(mktemp -d) || exit 1
RECTXT="$TMPDIR/vim.recovery.$USER.txt"
RECFN="$TMPDIR/vim.recovery.$USER.fn"
trap 'rm -f "$RECTXT" "$RECFN"; rmdir "$TMPDIR"' 0 1 2 3 15
for q in $SWAP_FILE_DIR/.*sw? $SWAP_FILE_DIR/*; do
  echo $q
  [[ -f $q ]] || continue
  rm -f "$RECTXT" "$RECFN"
  vim -X -r "$q" \
      -c "w! $RECTXT" \
      -c "let fn=expand('%')" \
      -c "new $RECFN" \
      -c "exec setline( 1, fn )" \
      -c w\! \
      -c "qa"
  if [[ ! -f $RECFN ]]; then
    echo "nothing to recover from $q"
    rm -f "$q"
    continue
  fi
  CRNT="$(cat $RECFN)"
  if [ "$CRNT" = "$RECTXT" ]; then
      echo "Can't find original file. Press enter to open vim so you can save the file. The swap file will be deleted afterward!"
      read
      vim "$CRNT"
      rm -f "$q"
  else if diff --strip-trailing-cr --brief "$CRNT" "$RECTXT"; then
      echo "Removing redundant $q"
      echo "  for $CRNT"
      rm -f "$q"
  else
      echo $q contains changes, or there may be no original saved file
      vim -n -d "$CRNT" "$RECTXT"
      rm -i "$q" || exit
  fi
  fi
done

Ответ 4

Отличный наконечник DiffOrig идеален. Вот bash script, который я использую для запуска в каждом файле подкачки в текущем каталоге:

#!/bin/bash

swap_files=`find . -name "*.swp"`

for s in $swap_files ; do
        orig_file=`echo $s | perl -pe 's!/\.([^/]*).swp$!/$1!' `
        echo "Editing $orig_file"
        sleep 1
        vim -r $orig_file -c "DiffOrig"
        echo -n "  Ok to delete swap file? [y/n] "
        read resp
        if [ "$resp" == "y" ] ; then
                echo "  Deleting $s"
                rm $s
        fi
done

Вероятно, можно было бы использовать еще некоторую проверку ошибок и цитирование, но работало до сих пор.

Ответ 5

Я предпочитаю не устанавливать рабочий каталог VIM в .vimrc. Здесь приведена модификация chouser script, которая копирует файлы подкачки на путь подкачки по запросу, проверяя дубликаты, а затем сверяет их. Это было написано, бросилось, не забудьте оценить его, прежде чем использовать его для практического использования.

#!/bin/bash

if [[ "$1" == "-h" ]] || [[ "$1" == "--help" ]]; then
   echo "Moves VIM swap files under <base-path> to ~/.vim/swap and reconciles differences"
   echo "usage: $0 <base-path>"
   exit 0
fi

if [ -z "$1" ] || [ ! -d "$1" ]; then
   echo "directory path not provided or invalid, see $0 -h"
   exit 1
fi

echo looking for duplicate file names in hierarchy
swaps="$(find $1 -name '.*.swp' | while read file; do echo $(basename $file); done | sort | uniq -c | egrep -v "^[[:space:]]*1")"
if [ -z "$swaps" ]; then
   echo no duplicates found
   files=$(find $1 -name '.*.swp')
   if [ ! -d ~/.vim/swap ]; then mkdir ~/.vim/swap; fi
   echo "moving files to swap space ~./vim/swap"
   mv $files ~/.vim/swap
   echo "executing reconciliation"
   TMPDIR=$(mktemp -d) || exit 1
   RECTXT="$TMPDIR/vim.recovery.$USER.txt"
   RECFN="$TMPDIR/vim.recovery.$USER.fn"
   trap 'rm -f "$RECTXT" "$RECFN"; rmdir "$TMPDIR"' 0 1 2 3 15
   for q in ~/.vim/swap/.*sw? ~/.vim/swap/*; do
     [[ -f $q ]] || continue
     rm -f "$RECTXT" "$RECFN"
     vim -X -r "$q" \
         -c "w! $RECTXT" \
         -c "let fn=expand('%')" \
         -c "new $RECFN" \
         -c "exec setline( 1, fn )" \
         -c w\! \
         -c "qa"
     if [[ ! -f $RECFN ]]; then
       echo "nothing to recover from $q"
       rm -f "$q"
       continue
     fi
     CRNT="$(cat $RECFN)"
     if diff --strip-trailing-cr --brief "$CRNT" "$RECTXT"; then
         echo "removing redundant $q"
         echo "  for $CRNT"
         rm -f "$q"
     else
         echo $q contains changes
         vim -n -d "$CRNT" "$RECTXT"
         rm -i "$q" || exit
     fi
   done
else
   echo duplicates found, please address their swap reconciliation manually:
   find $1 -name '.*.swp' | while read file; do echo $(basename $file); done | sort | uniq -c | egrep '^[[:space:]]*[2-9][0-9]*.*'
fi

Ответ 6

У меня есть это в моем .bashrc файле. Я хотел бы дать соответствующий кредит части этого кода, но я забыл, откуда я его получил.

mswpclean(){

for i in `find -L -name '*swp'`
do
    swpf=$i
    aux=${swpf//"/."/"/"}
    orif=${aux//.swp/}
    bakf=${aux//.swp/.sbak}

    vim -r $swpf -c ":wq! $bakf" && rm $swpf
    if cmp "$bakf" "$orif" -s
    then rm $bakf && echo "Swap file was not different: Deleted" $swpf
    else vimdiff $bakf $orif
    fi
done

for i in `find -L -name '*sbak'`
do
    bakf=$i
    orif=${bakf//.sbak/}
    if test $orif -nt $bakf
    then rm $bakf && echo "Backup file deleted:" $bakf
    else echo "Backup file kept as:" $bakf
    fi
done }

Я просто запускаю это в корне моего проекта и, если файл отличается, он открывает vim diff. Затем будет сохранен последний сохраненный файл. Чтобы сделать его идеальным, мне просто нужно было бы заменить последнее:

else echo "Backup file kept as:" $bakf

чем-то вроде

else vim $bakf -c ":wq! $orif" && echo "Backup file kept and saved as:" $orif

но я не успел его проверить.

Надеюсь, что это поможет.