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
    • 最新
    • 标签
主页 / unix / 问题 / 787151
Accepted
jnrbsn
jnrbsn
Asked: 2024-11-25 09:26:44 +0800 CST2024-11-25 09:26:44 +0800 CST 2024-11-25 09:26:44 +0800 CST

为什么 cron 运行的 bash 脚本中的 exec 不保留 $PATH?

  • 772

我在 Debian 12 上设置了以下 cron 作业:

/etc/cron.d/jonathan-test:

SHELL=/bin/bash
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin

* * * * * jonathan /home/jonathan/test1.sh >> /home/jonathan/test.log 2>&1

/home/jonathan/test1.sh:

#!/usr/bin/env bash

export PATH="/home/jonathan/mytestdir:${PATH}"
echo "test1.sh -> PATH=${PATH}"
export PAAATH="this_is_a_test"
echo "test1.sh -> PAAATH=${PAAATH}"
exec "${HOME}/test2.sh"

/home/jonathan/test2.sh:

#!/usr/bin/env bash

echo "test2.sh -> PATH=${PATH}"
echo "test2.sh -> PAAATH=${PAAATH}"

当它运行时,它会将以下内容写入/home/jonathan/test.log:

test1.sh -> PATH=/home/jonathan/mytestdir:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
test1.sh -> PAAATH=this_is_a_test
test2.sh -> PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
test2.sh -> PAAATH=this_is_a_test

如您所见,$PATH变量未被保留exec。

这是我实际问题的一个简化示例,即从 cron 作业运行pyenvcron.d 。如果我将文件更改为以下内容:

SHELL=/bin/bash
PYENV_ROOT=/opt/pyenv
PATH=/opt/pyenv/shims:/opt/pyenv/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin

* * * * * jonathan python --version >> /home/jonathan/test.log 2>&1

然后我将其写入输出文件:

/opt/pyenv/libexec/pyenv-exec: line 24: pyenv-version-name: command not found

它正确执行了/opt/pyenv/shims/python。那只是一个运行的 bash 脚本pyenv exec python --version。它正确执行了/opt/pyenv/bin/pyenv,这是一个指向的符号链接/opt/pyenv/libexec/pyenv,这是一个修改$PATH为包含的bash 脚本/opt/pyenv/libexec(是的,它确实导出了它!)并执行了/opt/pyenv/libexec/pyenv-exec,这是另一个 bash 脚本,它试图PYENV_VERSION="$(pyenv-version-name)"在第 24 行执行,这导致了上面的错误,因为/opt/pyenv/libexec不在$PATH。我把它缩小到上面的简化示例。当不从 cron 运行时,仅使用环境变量且没有 shell 集成的完全相同的pyenv设置也可以正常工作。

仅供参考,这里没有sudo任何地方,我也可以像其他用户一样重现它。所以它似乎与secure_path无关/etc/sudoers。

bash
  • 1 1 个回答
  • 71 Views

1 个回答

  • Voted
  1. Best Answer
    jnrbsn
    2024-11-27T23:30:11+08:002024-11-27T23:30:11+08:00

    既然我已经明白了,我就来回答我自己的问题。

    问题是我有一$BASH_ENV组自定义的。当以非交互方式调用时,bash 不会读取.profile、.bash_profile或.bashrc(文档)。这是 cron 作业的常见问题,因为它们不会具有许多人期望的环境。但是,您可以使用$BASH_ENV。当以非交互方式调用 bash 时,它的行为就像它这样做了:

    if [[ -n "$BASH_ENV" ]]; then source "$BASH_ENV"; fi
    

    因此,为了获得一个一致的环境(无论我的脚本是否从 cron 运行),我指向了$BASH_ENV一个如下所示的文件:

    source /etc/profile
    source ~/.profile
    

    因此,我的 cron 作业将获得与从登录 shell 调用相同的环境。这有两个问题。首先,/etc/profile是定义 $PATH而不是添加。其次,我没有意识到$BASH_ENV为每个子 shell 加载,所以我的$PATHcron 作业不断为每个子 shell 重置。

    因此,我将其更改/etc/profile为检查特定路径$PATH,如果不存在则添加它们。我将$BASH_ENV文件更改为:

    # if this file was already loaded, don't do anything
    [[ -n "$BASH_ENV_LOADED" ]] && return
    
    # load the environment that a login shell would normally get
    source /etc/profile
    source ~/.profile
    
    # indicate that this file has been loaded
    export BASH_ENV_LOADED=1
    

    (我可能不应该使用以 开头的变量名BASH_,但我想不出该怎么称呼它。)

    • 3

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

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

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

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

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve