如何在vi文本编辑器中获得文件的sudo访问权限?

通常在编辑配置文件时,我会用vi打开一个,然后当我保存它时,我意识到我没有输入

sudo vi filename

有没有办法给vi sudo特权来保存文件?我好像记得之前在查vi的时候看到过这个,但是现在找不到了。

80869 次浏览

一个快速谷歌似乎给出了这样的建议:

  1. 如果它是只读的,不要尝试编辑。
  2. 您可以更改文件的权限。(它是否能让你省钱取决于试验。)
  3. 如果你仍然编辑,保存到一个临时文件,然后移动它。

http://ubuntuforums.org/showthread.php?t=782136

Ryan的建议通常是好的,但是,如果遵循步骤3,不要移动临时文件;它将拥有错误的所有权和权限。相反,sudoedit是正确的文件,并读入临时文件的内容(使用:r或类似的方法)。

如果执行步骤2,使用:w!强制写入文件。

一般来说,你不能改变vi进程的有效用户id,但是你可以这样做:

:w !sudo tee myfile

您可以考虑的一个快速hack是对正在编辑的文件执行chmod,用vim保存,然后chmod到文件的原始状态。

ls -l test.file (to see the permissions of the file)
chmod 777 test.file
[This is where you save in vim]
chmod xxx test.file (restore the permissions you found in the first step)

当然,我不建议在您担心安全性的系统中使用这种方法,因为在几秒钟内任何人都可以在您不知情的情况下读取/更改文件。

当您进入需要sudo访问权限来编辑的文件的插入模式时,您会得到一条状态消息

-- INSERT -- W10: Warning: Changing a readonly file

如果我错过了,通常我会错过

:w ~/edited_blah.tmp
:q

. .然后. .

sudo "cat edited_blah.tmp > /etc/blah"

…或…

sudo mv edited_blah.tmp /etc/blah

也许有一种不那么迂回的方法,但它确实有效。

%被替换为当前文件名,因此你可以使用:

:w !sudo tee %

(vim将检测到文件已被更改,并询问您是否希望重新加载它。通过选择[L]而不是OK来确定。)

作为快捷方式,您可以定义自己的命令。在你的.vimrc中放入以下内容:

command W w !sudo tee % >/dev/null

使用上述方法,你可以输入:W<Enter>来保存文件。自从我写了这篇文章,我找到了一个更好的方法来做到这一点:

cmap w!! w !sudo tee >/dev/null %

通过这种方式,你可以输入:w!!,它将被扩展到完整的命令行,将光标留在末尾,所以如果你愿意,你可以用你自己的文件名替换%

如果你正在使用Vim,有一个名为sudo.vim的脚本可用。如果你发现你打开了一个需要根权限来读取的文件,type

: e sudo: %
Vim用当前文件的名称替换%,并且sudo:指示sudo。Vim脚本接管读写。

我在~/.bashrc中有这个:

alias svim='sudo vim'

现在,每当我需要编辑配置文件时,我只需用svim打开它。

这是另一个在回答这个问题后出现的插件,一个名为SudoEdit的插件,它提供SudoRead和SudoWrite函数,默认情况下将首先尝试使用sudo,如果失败则尝试使用su: http://www.vim.org/scripts/script.php?script_id=2709

常见的警告

绕过只读文件问题的最常见方法是作为超级用户使用sudo tee的实现打开一个通往当前文件的管道。然而,我在互联网上找到的所有最流行的解决方案都有几个潜在的警告:

  • 整个文件和文件都被写入终端。对于大文件,这可能很慢,特别是在较慢的网络连接上。
  • 文件将丢失其模式和类似属性。
  • 包含不寻常字符或空格的文件路径可能无法正确处理。

解决方案

要解决所有这些问题,您可以使用以下命令:

" On POSIX (Linux/Mac/BSD):
:silent execute 'write !sudo tee ' . shellescape(@%, 1) . ' >/dev/null'


" Depending on the implementation, you might need this on Windows:
:silent execute 'write !sudo tee ' . shellescape(@%, 1) . ' >NUL'

可以缩写为:

:sil exec 'w !sudo tee ' . shellescape(@%, 1) . ' >/dev/null'
:sil exec 'w !sudo tee ' . shellescape(@%, 1) . ' >NUL'

解释

:开始命令;您需要在普通模式下键入此字符以开始输入命令。在脚本中应该省略。

sil[ent]抑制命令的输出。在本例中,我们希望停止运行:!命令后出现的类似__abc1的提示符。

exec[ute]将字符串作为命令执行。我们不能只运行:write,因为它不会处理必要的函数调用。

!表示:!命令::write接受的唯一命令。通常,:write接受要写入的文件路径。:!单独在shell中运行命令(例如,使用bash -c)。使用:write,它将在shell中运行命令,然后将整个文件写入stdin

sudo应该是显而易见的,因为这就是你在这里的原因。以超级用户身份执行该命令。网上有很多关于这是如何运作的信息。

teestdin输送到给定的文件。:write将写入stdin,然后超级用户tee将接收文件内容并写入文件。它不会创建一个新文件——只是覆盖内容——因此文件模式和属性将被保留。

shellescape()转义指定文件路径中的特殊字符,适用于当前shell。如果只有一个参数,它通常会在必要时用引号将路径括起来。由于我们要发送到一个完整的shell命令行,我们将希望传递一个非零值作为第二个参数,以启用其他特殊字符的反斜杠转义,否则可能会使shell出错。

@%读取%寄存器的内容,其中包含当前缓冲区的文件名。它不一定是一个绝对路径,所以请确保您没有更改当前目录。在某些解决方案中,您会看到commercial-at符号被省略。根据位置的不同,%是一个有效的表达式,并且与读取%寄存器具有相同的效果。嵌套在另一个表达式中的快捷方式通常是不允许的:例如在这种情况下。

>NUL>/dev/nullstdout重定向到平台的空设备。即使我们已经禁用了该命令,我们也不希望所有与将stdin输送回vim相关的开销——最好尽早转储它。NUL是DOS, MS-DOS和Windows上的空设备,不是有效的文件。在Windows 8中,重定向到NUL不会导致写入名为NUL的文件。尝试在桌面上创建一个名为NUL的文件,带或不带文件扩展名:您将无法这样做。(Windows中还有其他几个设备名称可能值得了解。)

~ / . vimrc

平台相关的

当然,你也不希望每次都把它们记下来。将适当的命令映射到更简单的用户命令要容易得多。要在POSIX上做到这一点,你可以在你的~/.vimrc文件中添加以下行,如果它不存在就创建它:

command W silent execute 'write !sudo tee ' . shellescape(@%, 1) . ' >/dev/null'

这将允许您键入:W(区分大小写)命令以超级用户权限写入当前文件——这要容易得多。

独立于平台

我使用了一个独立于平台的~/.vimrc文件来进行跨计算机同步,所以我在我的文件中添加了多平台功能。下面是一个只有相关设置的~/.vimrc:

#!vim
" Use za (not a command; the keys) in normal mode to toggle a fold.
" META_COMMENT Modeline Definition: \{\{{1
" vim: ts=4 sw=4 sr sts=4 fdm=marker ff=unix fenc=utf-8
"   ts:     Actual tab character stops.
"   sw:     Indentation commands shift by this much.
"   sr:     Round existing indentation when using shift commands.
"   sts:    Virtual tab stops while using tab key.
"   fdm:    Folds are manually defined in file syntax.
"   ff:     Line endings should always be <NL> (line feed #09).
"   fenc:   Should always be UTF-8; #! must be first bytes, so no BOM.




" General Commands: User Ex commands. \{\{{1
command W call WriteAsSuperUser(@%)         " Write file as super-user.




" Helper Functions: Used by user Ex commands. \{\{{1
function GetNullDevice() " Gets the path to the null device. \{\{{2
if filewritable('/dev/null')
return '/dev/null'
else
return 'NUL'
endif
endfunction


function WriteAsSuperUser(file) " Write buffer to a:file as the super user (on POSIX, root). \{\{{2
exec '%write !sudo tee ' . shellescape(a:file, 1) . ' >' . GetNullDevice()
endfunction




" }}}1
" EOF