AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / ubuntu / 问题 / 980118
Accepted
James Newton
James Newton
Asked: 2017-11-26 05:40:22 +0800 CST2017-11-26 05:40:22 +0800 CST 2017-11-26 05:40:22 +0800 CST

bash 脚本:使用或不使用 sudo 调用时的不同结果

  • 772

在 Ubuntu 16.04.3 中,我有一个非常简单的 bash 脚本:

测试.sh

[[ 0 == 0 ]] && result="true" || result="false"
echo $result
echo $USER $SHELL $0

当我将它称为非 root 用户me或 asroot时,它按预期工作。如果我使用sudo ./test.sh,它会抱怨语法错误:

$ ./test.sh
true
me /bin/bash ./test.sh

$ sudo su
# ./test.sh 
true
root /bin/bash ./test.sh

# exit
$ sudo ./test.sh
./test.sh: 1: ./test.sh: [[: not found
false
root /bin/bash ./test.sh

这可能是什么原因造成的?我怎样才能修复它,以便me可以正常使用这个脚本和 with sudo?

bash
  • 2 2 个回答
  • 2098 Views

2 个回答

  • Voted
  1. Best Answer
    dessert
    2017-11-26T05:50:05+08:002017-11-26T05:50:05+08:00

    每个脚本都以Shebang开头,没有它,启动脚本的 shell 不知道应该由哪个解释器运行您的脚本1并且可能 - 就像sudo ./script.sh这里的情况一样 - 运行它sh,在 Ubuntu 16.04 中链接到dash. 条件表达式 [[是一个bash复合命令,所以dash不知道如何处理并抛出你遇到的错误。

    这里的解决方案是添加

    #!/bin/bash
    

    作为脚本的第一行。当您使用 明确调用它时,您可能会得到相同的结果sudo bash ./script.sh,但是 shebang 是要走的路。
    要检查哪个 shell 运行您的脚本,请添加echo $0到它。这与 echo $SHELL引用wiki.archlinux.org的情况不同:

    SHELL 包含用户首选 shell 的路径。请注意,这不一定是当前正在运行的 shell,尽管 Bash 在启动时设置了这个变量。

    1:正如您刚开始假设./test.sh的那样,子shell也是如此。bashbashsudo su

    • 20
  2. terdon
    2017-11-26T09:07:02+08:002017-11-26T09:07:02+08:00

    正如@dessert 解释的那样,这里的问题是您的脚本没有shebang 行。如果没有 shebang,sudo将默认尝试使用/bin/sh. 我在任何地方都找不到它的文档,但我通过检查在sudo文件中找到以下内容的源代码进行了确认pathnames.h:

    #ifndef _PATH_BSHELL
    #define _PATH_BSHELL "/bin/sh"
    #endif /* _PATH_BSHELL */
    

    这意味着“如果_PATH_BSHELL未定义变量,请将其设置为/bin/sh”。然后,在configure源 tarball 中包含的脚本中,我们有:

    for p in "/bin/bash" "/usr/bin/sh" "/sbin/sh" "/usr/sbin/sh" "/bin/ksh" "/usr/bin/ksh" "/bin/bash" "/usr/bin/bash"; do
        if test -f "$p"; then
        found=yes
        { $as_echo "$as_me:${as_lineno-$LINENO}: result: $p" >&5
    $as_echo "$p" >&6; }
        cat >>confdefs.h <<EOF
    #define _PATH_BSHELL "$p"
    EOF
    
        break
        fi
    done
    

    此循环将查找/bin/bash、/usr/bin/sh、或/sbin/sh,然后将 设置为首先找到的那个。由于是列表中的第一个并且它存在,因此设置为. 所有这一切的结果是,除非另有定义,否则的默认 shell是./usr/sbin/sh/bin/ksh_PATH_BSHELL/bin/sh_PATH_BSHELL/bin/shsudo/bin/sh

    因此,sudo将默认使用/bin/sh和在 Ubuntu 上运行东西,这是一个符号链接dash,一个最小的 POSIX 兼容 shell:

    $ ls -l /bin/sh
    lrwxrwxrwx 1 root root 4 Feb 27  2015 /bin/sh -> dash
    

    该[[构造是一个 bash 功能,它不是由 POSIX 标准定义的,也不被理解dash:

    $ bash -c '[[ true ]] && echo yes'
    yes
    $ dash -c '[[ true ]] && echo yes'
    dash: 1: [[: not found
    

    详细地说,在您尝试的三个调用中:

    1. ./test.sh

      没有sudo;在没有 shebang 行的情况下,您的 shell 将尝试执行文件本身。由于您正在运行bash,这将有效地运行bash ./test.sh和工作。

    2. sudo su其次是./test.sh。

      在这里,您正在为 user 启动一个新的 shell root。这将是在$SHELL环境变量中为该用户定义的任何 shell,在 Ubuntu 上,root 的默认 shell 是bash:

      $ grep root /etc/passwd
      root:x:0:0:root:/root:/bin/bash
      
    3. sudo ./test.sh

      在这里,您让sudo直接执行命令。由于它的默认 shell/bin/sh如上所述,这会导致它使用 运行脚本/bin/sh,这是dash因为dash不理解而失败[[。


    注意:如何设置默认 shell 的细节sudo似乎有点复杂。我尝试将答案中提到的文件更改为指向/bin/bash但sudo仍默认为/bin/sh. 所以源代码中肯定还有其他一些地方定义了默认的shell。尽管如此,主要观点(sudo默认为sh)仍然存在。

    • 5

相关问题

  • 同时复制到两个位置

  • 如何在 shell 脚本中创建选择菜单?

  • 从 bash 迁移到 zsh [关闭]

  • bashrc 还是 bash_profile?

  • 备份 bash 脚本未压缩其 tarball

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何运行 .sh 脚本?

    • 16 个回答
  • Marko Smith

    如何安装 .tar.gz(或 .tar.bz2)文件?

    • 14 个回答
  • Marko Smith

    如何列出所有已安装的软件包

    • 24 个回答
  • Marko Smith

    无法锁定管理目录 (/var/lib/dpkg/) 是另一个进程在使用它吗?

    • 25 个回答
  • Martin Hope
    Flimm 如何在没有 sudo 的情况下使用 docker? 2014-06-07 00:17:43 +0800 CST
  • Martin Hope
    Ivan 如何列出所有已安装的软件包 2010-12-17 18:08:49 +0800 CST
  • Martin Hope
    La Ode Adam Saputra 无法锁定管理目录 (/var/lib/dpkg/) 是另一个进程在使用它吗? 2010-11-30 18:12:48 +0800 CST
  • Martin Hope
    David Barry 如何从命令行确定目录(文件夹)的总大小? 2010-08-06 10:20:23 +0800 CST
  • Martin Hope
    jfoucher “以下软件包已被保留:”为什么以及如何解决? 2010-08-01 13:59:22 +0800 CST
  • Martin Hope
    David Ashford 如何删除 PPA? 2010-07-30 01:09:42 +0800 CST

热门标签

10.10 10.04 gnome networking server command-line package-management software-recommendation sound xorg

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve