经常要在bash脚本里面或者直接对脚本本身加上sudo运行命令,但是这引发了一系列的问题。
比如用sudo的时候,脚本里的~或$HOME指代用户文件夹的这个变量,到底是应该指向我真正的用户文件夹如/home/pi呢,还是指向了超级管理员的用户文件夹/root/呢?
实际上它指向了/root/文件夹,这是我们绝对不想要的。但是很多命令如安装个程序,都不得不用sudo,那怎么办?
首先要说下经验:命令行的权限执行,从表现上来看,可以分为以下5种情况:
很多变量、环境变量在这4中情况下,会经常出现混乱!(混乱指的是我们自己,不是电脑)
另外,说个小技巧。
我们都直到~变量是指向当前用户目录,实际上~abc格式的变量可以指向指定用户的用户目录,如~pi会指向/home/pi,或~ubuntu指向/home/ubuntu.
理清一下思路:
在正常执行脚本如./test.sh时是没有任何问题的,即使脚本里面出现了sudo如sudo apt-get update这样也是没有问题的。
也就是说,就只有对整个脚本执行sudo的情况下如sudo ./test.sh,才会出现严重问题的!
那么假设我的真实用户是pi,而HOME目录在/home/pi,现在我要在sudo ./test.sh这样的执行方式下找出正确的解决方案。
以下为脚本中的各种语句和变量以及显示结果:
# (不推荐!) $ whoami >>> root # 不同于whoami,能够指出当前有哪些用户登录电脑,包括本机登录和ssh登录的所有人 $ who am i >>> 有些机器上显示为空 >>> Mac上显示: pi ttys001 Nov 26 16:57 # 等同于whoami (不推荐!) $ echo $USER >>> root # 用户主目录位置 (不靠谱不推荐!) echo $HOME >>> /root $ 用户主目录位置,等同于$HOME (不推荐!) $ echo ~ >>> /root # 直接使用环境变量LOGNAME $ echo $LOGNAME >>> root # 显式调用环境变量LOGNAME $ printenv LOGNAME >>> root # SUDO_USER是root的ENV中的环境变量, # 同时普通用户的env是没有的,只有root用户才能显示出来 $ sudo echo $SUDO_USER >>> pi # 显示调用环境变量SUDO_USER (不推荐!) # 从结果中可以看到,即使是sudo身份执行的脚本,脚本里面是否加sudo也会不同! $ printenv SUDO_USER >>> pi $ sudo printenv SUDO_USER >>> root
从上面测试中可以看出,如果我们是用sudo执行bash脚本的话,很多变量都是“不靠谱”的。
Stackoverflow中,比较一致性的倾向就是使用$SUDO_USER这个环境变量。而测试中也的确,它是最“稳定的”,即在不同的权限、OS系统下,都能始终如一(只限有sudo的系统)。
那么现在我们有了用户名,就可以用~pi这样的命令获取主目录/home/pi了,但是!
这时候问题又出现了:手敲时候,我们可以获得~pi的正确地址,但是脚本中却不识别~pi是个什么东西,顶多是个字符串,没法像变量一样。
那既然是这样,我们就不能用~abc方法了,改用虽然老套但是绝对不混乱的方法:
从/etc/passwd中直接看。
手动的话可以直接打开passwd查看,脚本里面就比较麻烦,最方便的是用系统命令getent即Get Entries命令,获得指定用户的信息:
$ getent passwd pi >>> pi:x:1000:1000:,,,:/home/pi:/bin/bash
那么,剩下的是有把其中的/home/pi取出来了,我们用cut就轻松取出。
所以全部过程如下:
me=$SUDO_USER myhome=`getent passwd $me | cut -d: -f 6`
顺利得到/home/pi!
再进一步,如果脚本没有以sudo方式运行呢?这时候root用户和普通用户的环境变量下都是没有SUDO_USER这个变量的。那么就需要加一步判断了:
me=${SUDO_USER:-$LOGNAME} myhome=`getent passwd $me | cut -d: -f 6`
即如果SUDO_USER为空,则正常使用$LOGNAME获取当前用户。为什么不用$USER而是用$LOGNAME呢?因为USER不是每个系统都有,但是LOGNAME是*nix系统下都会有的。
更新
由于部分OS不能正确获取LOGNAME,所以统一采用uid的方式获取用户路径:
HOUSE=`getent passwd ${SUDO_UID:-$(id -u)} | cut -d: -f 6`
再更新
MacOS没有/etc/passwd,也不支持getent passwd <UID>方式获取用户信息,但是sudo下也能保持$USER和$HOME变量内容不变。
所以更改为下:
HOUSE=${$(`getent passwd ${SUDO_UID:-$(id -u)} | cut -d: -f 6`):-$HOME}
即如果getent方式无法获取内容,则直接取$HOME的值。
再再更新
因为bash不支持以上嵌套的三元运算表达式,所以要拆开:
HOUSE="`cat /etc/passwd |grep ${SUDO_UID:-$(id -u)} | cut -d: -f 6`" HOUSE=${HOUSE:-$HOME}
再再再更新
如果是root的话,grep uid的时候会匹配到passwd中所有含0的行,所以要改进为以下:
HOUSE="`cat /etc/passwd |grep ^${SUDO_USER:-$(id -un)}: | cut -d: -f 6`" HOUSE=${HOUSE:-$HOME}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
13.4 使用者身份切换 什么?在 Linux 系统当中还要作身份的变换?这是为啥?可能有下面几个原因啦! 使用一般帐号:系统平日操作的好习惯 事实上,为了安全的缘故,一些老人家都会建议你,尽量以一般身份使用者来操作 Linux 的日常作业!等到需要设置系统环境时, 才变换身份成为 root 来进行系统管理,相对比较安全啦!避免作错一些严重的指令,例如恐怖的“ rm -rf / ”(千万作不得!)
问题内容: 我正在使用Python在系统级别的Linux中进行项目。因此,我想知道,如果我以普通用户身份运行代码,并且正在访问系统文件,那么它应该具有root权限,那么我如何提示输入root密码并以超级用户身份运行其他代码。我想知道,如何以超级用户身份以密码提示运行python脚本。 任何帮助将不胜感激。先感谢您.. 问题答案: 您可以做的另一件事是,如果脚本不是以root身份执行的,则脚本会自动
本文向大家介绍JavaScript脚本判断蜘蛛来源的方法,包括了JavaScript脚本判断蜘蛛来源的方法的使用技巧和注意事项,需要的朋友参考一下 今天介绍的这个JS判断蜘蛛来源的方法脚本是写在body的onload里面的。即页面加载时进行判断。代码如下: 这个JS判断蜘蛛来源方法不太好用。特别是在需要判断更多蜘蛛来源时代码写起来比较麻烦。
本文向大家介绍iOS判断身份证号码是否正确的方法,包括了iOS判断身份证号码是否正确的方法的使用技巧和注意事项,需要的朋友参考一下 ①根据百度百科中身份证号码的标准实现该方法 ②该方法只能判断18位身份证,且不能判断身份证号码和姓名是否对应(要看姓名和号码是否对应,应该有大量的数据库做对比才能实现) ③直接copy这段代码,就能通过调用这个方法判断身份证号码是否符合标准,非常easy 以上就是本文
本文向大家介绍使用python脚本自动生成K8S-YAML的方法示例,包括了使用python脚本自动生成K8S-YAML的方法示例的使用技巧和注意事项,需要的朋友参考一下 1、生成 servie.yaml 1.1、yaml转json service模板yaml 转成json的结构 1.2、关键代码 2、生成 deployment.yaml 2.1、yaml转json deployment模板yam
我正在尝试以root用户身份运行脚本。这是我的密码 它在出错 我也遵循了这些方法,但最终出现了错误