AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / unix / Perguntas / 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

Por que exec no script bash executado pelo cron não preserva $PATH?

  • 772

Tenho a seguinte configuração de tarefa cron no Debian 12:

/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}"

Quando executado, ele grava o seguinte em /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

Como você pode ver, a $PATHvariável não é preservada por exec.

Este é um exemplo simplificado e artificial do meu problema real, que é executar o pyenv a partir de um cron job. Se eu mudar meu cron.darquivo para isto:

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

Então eu escrevo isso no arquivo de saída:

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

Ele executa corretamente /opt/pyenv/shims/python. Isso é apenas um script bash que executa pyenv exec python --version. Ele executa corretamente /opt/pyenv/bin/pyenv, que é um link simbólico para /opt/pyenv/libexec/pyenv, que é um script bash que modifica $PATHpara incluir /opt/pyenv/libexec(e sim, ele exporta!) e executa /opt/pyenv/libexec/pyenv-exec, que é outro script bash que tenta fazer PYENV_VERSION="$(pyenv-version-name)"na linha 24, o que resulta no erro acima, porque /opt/pyenv/libexecnão está em $PATH. Eu reduzi para o exemplo simplificado acima. A mesma pyenvconfiguração exata com apenas variáveis ​​de ambiente e sem a integração do shell funciona muito bem quando não é executada a partir do cron.

Para sua informação, não há sudolugar nenhum nisso, e eu posso reproduzi-lo como outros usuários também. Então não parece estar relacionado a secure_pathem /etc/sudoers.

bash
  • 1 1 respostas
  • 71 Views

1 respostas

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

    Vou responder minha própria pergunta, já que eu descobri.

    O problema era que eu tinha um $BASH_ENVconjunto personalizado. Quando invocado de forma não interativa , o bash não lê .profile, .bash_profile, ou .bashrc( docs ). Este é um problema comum com tarefas cron porque elas não terão o ambiente que muitas pessoas esperam. No entanto, você pode usar $BASH_ENV. Quando o bash é invocado de forma não interativa, ele se comporta como se fizesse isso:

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

    Então, em uma tentativa de ter um ambiente consistente, independentemente de meus scripts serem executados no cron ou não, apontei $BASH_ENVpara um arquivo que se parecia com este:

    source /etc/profile
    source ~/.profile
    

    Então, meus trabalhos cron teriam o mesmo ambiente como se fossem invocados de um shell de login. Havia dois problemas com isso. Primeiro, /etc/profileestava definindo $PATH em vez de adicionar a ele. Segundo, eu não percebi que $BASH_ENVé carregado para cada sub-shell, então meu $PATHestava constantemente sendo redefinido para cada sub-shell.

    Então, eu mudei meu /etc/profilepara verificar caminhos específicos $PATHe adicioná-los se eles não estiverem lá. E eu mudei meu $BASH_ENVarquivo para isto:

    # 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
    

    (Eu provavelmente não deveria usar um nome de variável que começa com BASH_, mas não consegui pensar em como chamá-lo.)

    • 3

relate perguntas

  • exportar variáveis ​​​​env programaticamente, via stdout do comando [duplicado]

  • Problema estranho ao passar variáveis ​​do arquivo de texto

  • Enquanto a linha lê mantendo os espaços de escape?

  • ordem de substituição de processos `te` e `bash`

  • Execute um script muito lento até que seja bem-sucedido

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Possível firmware ausente /lib/firmware/i915/* para o módulo i915

    • 3 respostas
  • Marko Smith

    Falha ao buscar o repositório de backports jessie

    • 4 respostas
  • Marko Smith

    Como exportar uma chave privada GPG e uma chave pública para um arquivo

    • 4 respostas
  • Marko Smith

    Como podemos executar um comando armazenado em uma variável?

    • 5 respostas
  • Marko Smith

    Como configurar o systemd-resolved e o systemd-networkd para usar o servidor DNS local para resolver domínios locais e o servidor DNS remoto para domínios remotos?

    • 3 respostas
  • Marko Smith

    apt-get update error no Kali Linux após a atualização do dist [duplicado]

    • 2 respostas
  • Marko Smith

    Como ver as últimas linhas x do log de serviço systemctl

    • 5 respostas
  • Marko Smith

    Nano - pule para o final do arquivo

    • 8 respostas
  • Marko Smith

    erro grub: você precisa carregar o kernel primeiro

    • 4 respostas
  • Marko Smith

    Como baixar o pacote não instalá-lo com o comando apt-get?

    • 7 respostas
  • Martin Hope
    user12345 Falha ao buscar o repositório de backports jessie 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl Por que a maioria dos exemplos do systemd contém WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky Como exportar uma chave privada GPG e uma chave pública para um arquivo 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll status systemctl mostra: "Estado: degradado" 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim Como podemos executar um comando armazenado em uma variável? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S Por que /dev/null é um arquivo? Por que sua função não é implementada como um programa simples? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 Como ver as últimas linhas x do log de serviço systemctl 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - pule para o final do arquivo 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla Por que verdadeiro e falso são tão grandes? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis Substitua a string em um arquivo de texto enorme (70 GB), uma linha 2017-12-30 06:58:33 +0800 CST

Hot tag

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

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve