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 / user-154557

domson's questions

Martin Hope
domson
Asked: 2019-02-18 12:47:57 +0800 CST

Armadilha de sinal do trabalho em segundo plano sem pressionar enter

  • 1

No Bash 5, me deparei com uma situação em que quero fazer o seguinte:

trap 'echo trapped!' USR1
while true; do kill -SIGUSR1 $$; sleep 1; done &

Mas eu tenho que pressionar enter no teclado toda vez que o sinal é enviado para fazer o código de trap continuar.

Na verdade, espero que a armadilha seja executada sem nenhuma interação com o teclado.

EDITAR:

Finalmente consegui trabalhar assim:

trap 'echo trapped!' USR1
while true; do kill -USR1 $$; kill -INT $$; sleep 1; done &

Mas considero a solução como solução alternativa e não como resposta.

bash shell-script
  • 1 respostas
  • 422 Views
Martin Hope
domson
Asked: 2019-02-16 07:06:54 +0800 CST

Como sinalizar uma armadilha no escopo "global" do bash da função?

  • 2

Eu quero executar um comando trap no escopo "global", mas o sinal virá de dentro da função. Claro que é possível declarar a variável globalmente de antemão ou usar a -gopção declare. Mas nos casos em que eu quero fonte na armadilha, isso não é muito praticável, como mostrado abaixo:

#!/bin/bash
# ./variables
declare say=hello
declare -i times=4

e o script real:

#!/bin/bash
# ./trapsource

setTraps () {
  trap 'echo trap called in scope ${FUNCNAME[@]}; source ./variables' SIGUSR1
}

sourceVariables () {
  kill -SIGUSR1 $$
}

setTraps
sourceVariables
echo I want you to $say \"hello\" $times times.
printf "$say\n%.0s" $(seq 1 $times)

Mas sem declare -g NAME, obrigado.

EDITAR:

É possível separar kill -s USR1 $$completamente a peça do processo atual, para que o sinal venha de 'fora'?

Tentei nohupe disownsem sucesso até agora.

PS: @LL2 ofereceu uma solução usando processo em segundo plano ou coproc aqui: https://unix.stackexchange.com/a/518720/154557 Mas temo que os problemas de sincronização de tempo apareçam e tornem o código desproporcionalmente complicado.

PPS: Na verdade, a solução @LL2 funciona muito bem com IPC, então considero a questão resolvida. Uma desvantagem é que os argumentos da função TÊM DE ser processados ​​dentro do arquivo de origem, se necessário, para uso posterior no escopo principal, porque é claro que os símbolos no subshell são perdidos. Dê uma olhada no EDIT 2 para outra solução para lidar com esse problema.

#!/bin/bash
# ./trapsource
set -x  # <-- set debugging to see what happens

setTraps () {
  trap 'declare IPC=./ipc.fifo; \
    [ -p $IPC ] \
    && exec {IPCFD}<>$IPC \
    && { read -a ARGS <&$IPCFD; eval "exec $IPCFD>&-"; } \
    && source "${ARGS[@]}" \
    && IPC_RCV=true' SIGUSR1
}

sourceVariables () {
  declare IPC=./ipc.fifo
  declare IPC_RCV=false
  [ ! -p $IPC ] \
  && mkfifo $IPC
  [ $# -gt 0 ] \
  && exec {IPCFD}<>$IPC \
  && echo "${@:2}" >&$IPCFD \
  && eval "exec $IPCFD>&-" \
  && kill -s USR1 $1
}

test () {
  sourceVariables "$@" &
}

setTraps
test $$ ./variables a b c
while ! [ $say ] ; do :; done  # <-- careful: cpu-intensive loop, only for 
demonstration
echo I want you to $say \"hello\" $times times.
printf "$say\n%.0s" $(seq 1 $times)
printf "'%s' " "${args[@]}"

EDIT 2 (Proposta 3 Trabalho em Andamento):

Alguém tem experiência suficiente com bash custom builtins para apontar em uma direção onde uma verdadeira solução pode ser alcançada?

PS: Bash Loadable Builtins são super poderosos e provavelmente podem ser usados ​​para receber chamadas de método dbus no escopo principal.

SOLUÇÃO II (sem bg nem coproc):

Parece que o sinal sempre é acionado dentro do escopo da linha de execução atual. Muito provavelmente existe outra solução mais sofisticada usando builtins bash personalizados (builtins carregáveis ​​que são bastante poderosos). Posso até pensar em bash-dbus-builtins para IPC. De qualquer forma, esta solução usando aliases pode ser útil se alguém quiser dizer a si mesmo/outro PID dormindo em segundo plano para fornecer algum script no sinal, sem ter que definir o sinalizador global -g no script de origem ./variables, então o ./ O script de variáveis ​​também pode ser usado exclusivamente dentro do escopo da função, se necessário. Como os aliases não podem ser exportados para subshells, seria necessário incluir o alias em um script de ambiente referenciado pela variável BASH_ENV, por exemplo. Aqui estendi o exemplo original com um mecanismo IPC para tornar o objetivo do caso de uso mais claro.

A variável store agora também recebe argumentos

#!/bin/bash
# ./variables
declare say=hello
declare -i times=4
declare -a args=("$@")

Primeiro eu preparo um script de ambiente

#!/bin/bash
# ./env
shopt -s expand_aliases
alias source_ipc='source ./ipc'
alias source_ipc_rcv='source ./ipc_rcv'

e certifique-se de disponibilizar o alias em cada script bash.

$> export BASH_ENV=./env

além disso, preciso do mecanismo de envio e reveive IPC real usando fifo

#!/bin/bash
# ./ipc

declare IPC=./ipc.fifo
declare IPC_RCV=false
[ ! -p $IPC ] \
&& mkfifo $IPC
[ $# -gt 0 ] \
&& exec {IPCFD}<>$IPC \
&& echo "${@:2}" >&$IPCFD \
&& eval "exec $IPCFD>&-" \
&& kill -s USR1 $1

mais ipc receber

#!/bin/bash
# ./ipc_rcv

declare IPC=./ipc.fifo
[ -p $IPC ] \
&& exec {IPCFD}<>$IPC \
&& { read -a ARGS <&$IPCFD; eval "exec $IPCFD>&-"; } \
&& source "${ARGS[@]}" \
&& IPC_RCV=true

Agora posso acionar de forma transparente o sinal de origem em vários processos

#!/bin/bash
# ./trapsource u+x

trap 'source_ipc_rcv' SIGUSR1

log="$0.log"
err="$0.err.log"
exec 1>$log
exec 2>$err

# test inside script
source_ipc $$ ./variables a b c

while true; do
  echo I want you to \"$say\" $times times.
  if $IPC_RCV; then
    printf "$say\n%.0s" $(seq 1 $times)
    printf "'%s' " "${args[@]}"
  fi
  sleep 1
done

De 'fora' do namespace do processo, o sinal pode ser acionado da seguinte maneira. Presumo que o Alias ​​já foi fonte por referência BASH_ENV.

$> ./trapsource & TSPID=$!; sleep 1; source_ipc $TSPID ./variables arg1 arg2; sleep 2; kill $TSPID
bash signals
  • 3 respostas
  • 1689 Views

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