蝴蝶效应--由'sudo -s ...'引发的vim autocmd使用异常
1. 背景介绍
自加入RedHat起,我就把家里的台式机(Ubuntu 16.04 LTS)的默认登录用户veli切换成了huanli, 主要是为了跟公司的电脑配置对齐以方便未来WFH,但引发了一个vim使用异常。在我的.vimrc中,有这样一段配置代码,
if has("autocmd")
" When editing a file, always jump to the last known cursor position.
autocmd BufReadPost *
\ if line("'\"") > && line("'\"") <= line("$") |
\ exe "normal g`\"" |
\ endif
endif " has("autocmd")
这段代码的作用就是记住前一次光标的位置,再次打开文件的时候自动跳转到那里。 这是一个非常有用的功能,对于喜欢使用vim的程序员来说。 当我切换成huanli用户后,这一功能莫名其妙的失灵了Orz。。。
2. 故障调试
201 - 基于vim不同版本的对比测试: vim版本应该是默认支持autocmd功能的,但为了排除是版本问题(当前版本是7.4.1689), 于是下载一个vim8.0的源码包,编译并安装到/usr/local, 发现问题依然存在。
202 - 基于不同host的对比测试: 在笔记本上使用同样版本的vim, 发现笔记本上工作正常,台式机上工作异常,于是判定vim版本(7.4.1689)没有问题,一定是台式机某种设置引发的异常。 操作步骤如下:
(1) 同时在笔记本和台式机的Terminal上用vim打开一个文件, e.g.
$ vim /tmp/foo.c
在vim里执行
autocmd BufReadPost *
比较分析输出,发现完全一致。 样例输出截图如下:
由此可见,在vim出问题的台式机上, autocmd是在正确工作的。 于是,问题的所在必然是autocmd产生的记录没有被保存下来。
203 - vim verbose输出分析,发现了关键文件.viminfo
$ vim -V /tmp/foo.c
chdir(/tmp)
fchdir() to previous dir
...<snip>...
Searching for "/usr/share/vim/vimfiles/after/pack/*/start/*"
Searching for "/home/huanli/.vim/after/pack/*/start/*"
not found in 'packpath': "pack/*/start/*"
Reading viminfo file "/home/huanli/.viminfo" info oldfiles
Press ENTER or type command to continue
因此,我们可以大胆地做如下猜测,autocmd在正常工作,上次的光标位置有被autocmd记录下来,但是因为未知的原因没有保存到.viminfo中去。
204 - 查看.viminfo的权限,发现owner:group是root:root(而不是huanli:huanli),于是找到了vim异常的root cause。
$ ls -l ~/.viminfo
-rw------- root root Jan : /home/huanli/.viminfo
显然,普通用户huanli没有权限修改.viminfo文件,所以autocmd记忆结果无法保存下来给下一次vim打开文件后使用。 解决的方法异常简单,
$ sudo chown huanli:huanli /home/huanli/.viminfo
但是,引起这次vim异常的罪魁祸首是什么呢?也就是说,为什么使用root做vim编辑的时候,.viminfo文件保存在/home/huanli目录下面而不是/root?
205 - 查看使用root用户时的环境变量HOME,发现HOME=/home/huanli而不是/root
root@DELL380:~# env | grep HOME
~HOME=/root
HOME=/home/huanli
好了,原来是HOME未被设置成/root, 虽然~HOME为/root。 于是,罪魁祸首找到了,那就是一个叫做so的alias。
huanli@DELL380:~$ alias | grep '~HOME'
alias so='sudo -s ~HOME=/root'
这跟我平常的使用习惯有关,因为我在普通用户(huanli)设置了彩色的PS1和alias so, 通过so就可以很容易地切换到root用户而且保持彩色的PS1。
原来如此,这就是典型的蝴蝶效应,一个由'sudo -s ~HOME=/root'引发的vim使用异常!!
3. 总结陈词
整个trouble shooting的过程其实是充满了乐趣的,虽然比较艰辛。我在结束202步的时候差一点就放弃了,因为要哄娃睡觉。后来冷静地想了想,估计跟存储有关(在Inspur干过存储还是有用的)。于是通过最为关键的'vim -V'找到了root cause(P.S. verbose输出对于软件调试简直就是必杀技)。软件调试和故障诊断其实跟医生看病差不多,基本思想是采用排除法。当然,也需要大胆地猜测,靠谱地猜测,经验越丰富,猜得越快,猜得越靠谱。医生越老越值钱,理论上说,程序员也是啊,呵呵。。。
It's not enough to be the best at what you do; you must be perceived as the only one who does what you do. | 在你那一行成为最强的还不够,你必须被视为你那一行的唯一。
蝴蝶效应--由'sudo -s ...'引发的vim autocmd使用异常的更多相关文章
- 引发了未经处理的异常:读取访问权限冲突。 _First 是 nullptr。
1.问题:程序崩溃出现错误 引发了未经处理的异常:读取访问权限冲突. _First 是 nullptr. string strreponse=0: 定义这条语句,字符串初始化错误. 自己开发了一个股票 ...
- 关于foreach中对集合执行Add或者Remove操作引发枚举值被修改异常
方法传入集合List<string> ids; 执行操作后再次循环引发异常 foreach (string id in ids) { ids.Add("a"); } 更 ...
- 一个低级错误引发Netty编码解码中文异常
前言 最近在调研Netty的使用,在编写编码解码模块的时候遇到了一个中文字符串编码和解码异常的情况,后来发现是笔者犯了个低级错误.这里做一个小小的回顾. 错误重现 在设计Netty的自定义协议的时候, ...
- 学习vim命令:“:w !sudo tee %”
学习vim命令:“:w !sudo tee %” Original URL:http://www.haw-haw.org/node/1501 原文来自于commandlinefu 原文是这样解释这个命 ...
- Debian下安装vim
问题描述:安装完系统以后,刚要打算开始写程序,发现,vim还没有装,用su -切换到root后 直接运行apt-get install vim,提示插入disc源,然后回车,陷入无法解决的状态. 上网 ...
- Ubuntu14 搭载vim环境查看源码
首先是下载完整的vim74,然后编译安装.遗憾的是当编译时,没有开启图形界面. 在安装新版本的Vim之前,你需要卸载原来安装的老版本Vim,依次在终端下执行下列命令: sudo apt-get rem ...
- Fedora 19 vim c语言开发环境
1. Fedora 19 居然没有自带 gcc 和 g++: sudo yum -y install gcc gcc-c++ 2. 安装 vim 和 cvim 插件: sudo yum -y vim ...
- vim/Gvim配置
" Sections:" -> General" -> VIM user interface" -> Colors and Fonts&quo ...
- vim配置-程序员【转】
Ubuntu11.10的vim升级后,版本为vi Improved 7.3.154功能很强大了.不过,程序员要根据自己的习惯配置好vimrc文件,是vim更加得心应手. 注:一般用户在自己的当前目录下 ...
随机推荐
- java web 中的WEB-INF文件夹
WEB-INF下的东西是禁止直接访问的.如果这个页面是你的,要想让人访问最好不要放在这个目录下.如果一定放在那里.你可以使用:request.getRequestDispatcher("/W ...
- linux系统编程之文件与IO(五):stat()系统调用获取文件信息
一.stat()获取文件元数据 stat系统调用原型: #include <sys/stat.h> int stat(const char *path, struct stat *buf) ...
- c++日志输出库 spdlog 简介(3)多线程控制台输出日志
spdlog源码分析:https://www.cnblogs.com/eskylin/p/6483199.html spdlog的异步模式使得spdLog可以支持多线程,于是写了一个多线程的小例子: ...
- NET 文件批量下载
HTML <a class="btn btn-warning" id="btnDownload">选中下载</a> JS /* 批量下载 ...
- C# 中数组、ArrayList、List<T> 区别
一:数组 //定义 ]; //赋值 strs[] = "A"; strs[] = "B"; //修改 strs[] = "C"; //取值 ...
- C#中的NameValueCollection简介
NameValueCollection继承自NameObjectCollectionBase,并且和一般的键值对不同的是,它支持集合中出现相同的Key. 引用:using System.Collect ...
- OpenvSwitch 解读
OpenvSwitch 解读 报文匹配流程参考下图 调用流程(内核): ovs_vport_receive->ovs_dp_process_received_packet->ovs_flo ...
- 使用VS Code开发.Net Core 2.0 MVC Web应用程序教程之三(配置文件读取)
干了一天的活,还有点时间,给兄弟们写点东西吧. 大家有没有发现一个问题?那就是在.Net Core的MVC项目里面,没有.config文件了!!!同志们,没有config文件了啊,这样搞,我以后要做些 ...
- 深入理解java内存模型
深入理解Java内存模型(一)——基础 深入理解Java内存模型(二)——重排序 深入理解Java内存模型(三)——顺序一致性 深入理解Java内存模型(四)——volatile 深入理解Java内存 ...
- 消息中间件(一)MQ详解及四大MQ比较
一.消息中间件相关知识 1.概述 消息队列已经逐渐成为企业IT系统内部通信的核心手段.它具有低耦合.可靠投递.广播.流量控制.最终一致性等一系列功能,成为异步RPC的主要手段之一.当今市面上有很多主流 ...