Script para silenciar um aplicativo

14

Meu objetivo é ser capaz de silenciar o aplicativo Spotify, não o sistema inteiro. Usando o comando: ps -C spotify -o pid=Eu consigo encontrar o ID do processo do Spotify, neste caso o ID é "22981". Com esse processo ID gostaria de pesquisar a partir desta lista: pacmd list-sink-inputs. Esse comando retorna uma lista como esta:

eric@eric-desktop:~$ pacmd list-sink-inputs
Welcome to PulseAudio! Use "help" for usage information.
>>> 1 sink input(s) available.
    index: 0
    driver: <protocol-native.c>
    flags: START_CORKED 
    state: RUNNING
    sink: 1 <alsa_output.pci-0000_00_1b.0.analog-stereo>
    volume: 0: 100% 1: 100%
            0: -0.00 dB 1: -0.00 dB
            balance 0.00
    muted: no
    current latency: 1019.80 ms
    requested latency: 371.52 ms
    sample spec: s16le 2ch 44100Hz
    channel map: front-left,front-right
                 Stereo
    resample method: (null)
    module: 8
    client: 10 <Spotify>
    properties:
        media.role = "music"
        media.name = "Spotify"
        application.name = "Spotify"
        native-protocol.peer = "UNIX socket client"
        native-protocol.version = "26"
        application.process.id = "22981"
        application.process.user = "eric"
        application.process.host = "eric-desktop"
        application.process.binary = "spotify"
        window.x11.display = ":0"
        application.language = "en_US.UTF-8"
        application.process.machine_id = "058c89ad77c15e1ce0dd5a7800000012"
        application.process.session_id = "058c89ad77c15e1ce0dd5a7800000012-1345692739.486413-85297109"
        application.icon_name = "spotify-linux-512x512"
        module-stream-restore.id = "sink-input-by-media-role:music"

Agora eu gostaria de correlacionar o application.process.id = "22981"para o índice de entrada do coletor, que neste caso é index: 0. Agora, com esse número de índice, eu precisaria executar este comando: pacmd set-sink-input-mute 0 1silenciar o Spotify e pacmd set-sink-input-mute 0 0silenciar o Spotify. Para esses comandos, o primeiro número precisaria ser substituído pelo número de índice encontrado anteriormente, e o próximo número será o booleano para ativar ou desativar o mudo. Como posso colocar isso em um script, para que eu possa obter um comando para silenciar / ativar o som do aplicativo Spotify?

Edição: Ambos os scripts abaixo funcionam conforme o esperado, alguém pode adicionar uma alternância que iria verificar muted: yesou muted: noe, em seguida, mudo ou mudo em conformidade?

Edição: Consegui modificar o script de glenn jackman para adicionar a alternância:

#!/bin/bash

main() {
    local action=toggle
    while getopts :mu option; do 
        case "$option" in 
            m) action=mute ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))
    local pid=$(pidof "$1")
    if [[ -z "$pid" ]]; then
        echo "error: no running processes for: $1" >&2
    elif [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name" 
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "Usage: $0 [-m | -u] appname"
    echo "Default: toggle mute"
    echo "Arguments:"
    echo "-m = mute application"
    echo "-u = unmute application"
    exit $1
}

toggle() {
    local status=$(get_status "$1")
    if [[ "$status" == "yes" ]]; then
      unmute "$1"
    elif [[ "$status" == "no" ]]; then
      mute "$1"
    fi
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() { 
    local index=$(get_index "$1")
    local status=$(get_status "$1")
    [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null 
}

get_index() {
    local pid=$(pidof "$1")
    pacmd list-sink-inputs | 
    awk -v pid=$pid '
    $1 == "index:" {idx = $2} 
    $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
    '
}

get_status() {
   local pid=$(pidof "$1")
   pacmd list-sink-inputs | 
   awk -v pid=$pid '
   $1 == "muted:" {idx = $2} 
   $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
   '
}

main "$@"
era878
fonte
por que não usar pactl list sink-inputs? então funcionará pela rede.
Janus Troelsen

Respostas:

13

Aqui está minha opinião sobre seu desafio interessante:

#!/bin/bash

main() {
    local action=mute
    while getopts :hu option; do 
        case "$option" in 
            h) usage 0 ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name" 
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute application (default action is to mute)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() { 
    local index=$(get_index "$1")
    [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null 
}

get_index() {
    local pid=$(pidof "$1")
    if [[ -z "$pid" ]]; then
        echo "error: no running processes for: $1" >&2
    else
        pacmd list-sink-inputs | 
        awk -v pid=$pid '
            $1 == "index:" {idx = $2} 
            $1 == "application.process.id" && $3 == "\"" pid "\"" {print idx; exit}
        '
    fi
}

main "$@"
Glenn Jackman
fonte
Isso funciona perfeitamente também
era878 26/08/12
@ era878, eu gosto da ideia de alternar como a ação padrão. No entanto, sua get_statusfunção encontrará apenas as linhas "silenciadas" sem verificar se o status pertence ao aplicativo apropriado. Releia minha get_indexfunção para obter detalhes.
Glenn Jackman
3
bom awk habilidades :)
hytromo
@glennjackman, Sim, eu descobri isso depois de um tempo. Acredito que o script que acabei de postar funcione corretamente agora.
precisa saber é o seguinte
1
Requisitos: awk -v var=val. O awk circula sobre as linhas, 1 por 1, tenta corresponder a qualquer uma das $1 == ...instruções, executa o código entre colchetes na correspondência e continua. A primeira instrução corresponde às linhas cuja primeira palavra é index:e armazena a segunda palavra (SINK INDEX) na idxvariável. Portanto, idxé substituído pela próxima index: <SINK INDEX>linha até que o awk corresponda à segunda instrução ( $1= application.process.id, $2= =, $3= <expected pid val>). Quando essa segunda instrução corresponde, o awk é impresso idx(que é a última linha que corresponde à primeira index:) e sai.
KrisWebDev 22/10
7

obrigado pela solução! Consegui usar os scripts fornecidos aqui para corrigir meu problema. Desde que eu tive que modificá-los um pouco, aqui vou me juntar à versão melhorada.

A razão pela qual os scripts originais não funcionaram para mim é porque alguns aplicativos podem ter várias instâncias, ou seja, vários PID, mas talvez apenas um deles esteja produzindo som e, portanto, esteja realmente conectado ao Pulseaudio. Como o script usava apenas o primeiro PID encontrado, normalmente / não / desativava o aplicativo desejado.

Então, aqui está uma versão em que o argumento é o nome do aplicativo, conforme registrado no PulseAudio. Você pode encontrar esse nome executando o pacmd list-sink-inputscomando e procure o application.namecampo.

Uma solução alternativa seria ativar / desativar o som de todos os PIDs com o mesmo nome de aplicativo.

#!/bin/bash

# Adapter from glenn jackman on http://askubuntu.com/questions/180612/script-to-mute-an-application
# to depend directly on the name of the PulseAudio client
# rather than the application name (several instances of one application could
# run while only one is connected to PulseAudio)

# Possible further improvement: it could be useful to also mute all clients having
# the specified name. Here, only the first one is muted.

#!/bin/bash

main() {
    local action=mute
    while getopts :hu option; do
        case "$option" in
            h) usage 0 ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify the name of a PulseAudio client"
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute application (default action is to mute)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }

adjust_muteness() {
    local index=$(get_index "$1")
    if [[ -z "$index" ]]; then
        echo "error: no PulseAudio sink named $1 was found" >&2
    else
        [[ "$index" ]] && pacmd set-sink-input-mute "$index" $2 >/dev/null
    fi
}

get_index() {
#    local pid=$(pidof "$1")
#    if [[ -z "$pid" ]]; then
#        echo "error: no running processes for: $1" >&2
#    else
        pacmd list-sink-inputs |
        awk -v name=$1 '
            $1 == "index:" {idx = $2}
            $1 == "application.name" && $3 == "\"" name "\"" {print idx; exit}
        '
#    fi
}

main "$@"
Entalhe
fonte
6

Mesmo que a pergunta esteja pedindo um script, eu queria deixar isso aqui.

Eu escrevi um aplicativo C que faz isso no Ubuntu. Melhor ainda, ele fica na bandeja do indicador (usando libappindicator) e verifica o que o Spotify está tocando, em intervalos curtos. Se estiver reproduzindo um anúncio (verifica uma lista negra), o Silenciar é desativado. Se um novo anúncio estiver sendo reproduzido, basta clicar em Mudo no menu do indicador e ele será adicionado à lista negra.

O que ele faz é procurar uma janela X, para a qual XFetchNameretorna Spotify - Linux Preview. Em seguida, chama XGetWindowPropertypara consultar a _NET_WM_ICON_NAMEpropriedade dessa janela, que retorna uma string no "Spotify – <Artist> – <Song>"formato. Ao reproduzir anúncios, ele retorna algo como isto:

"Spotify – Spotify – Premium Free Trial Cancel Any Time"

Ele mantém uma Árvore de Pesquisa Ternária da lista de anúncios, para verificar com eficiência se o título atual está na lista.

Ele também usa a API assíncrona PulseAudio para consultar sink-inputse set-mute:

pa_context_get_sink_input_info_list()
pa_context_set_sink_input_mute()

Como é apenas um código C simples, é leve. Confira o código fonte e o .debpacote Ubuntu em: indicator-muteads . Provavelmente venceria um script de shell em 2-3 ordens de magnitude.

mkayaalp
fonte
não funciona com a versão 1.0.11
Janus Troelsen
4

Primeiro, a maneira "mais correta" de encontrar o PID de um aplicativo, como o spotify, é usar:

pidof spotify

Criei um script que faz o trabalho, não sei se é a melhor maneira de fazê-lo, mas funciona perfeitamente:

#!/bin/bash
# Script to mute an application using PulseAudio, depending solely on
# process name, constructed as answer on askubuntu.com: 
# http://askubuntu.com/questions/180612/script-to-mute-an-application

#It works as: mute_application.sh vlc mute OR mute_application.sh vlc unmute

if [ -z "$1" ]; then
   echo "Please provide me with an application name"
   exit 1
fi

if [ -z "$2" ]; then
   echo "Please provide me with an action mute/unmute after the application name"
   exit 1
fi

if ! [[ "$2" == "mute" || "$2" == "unmute" ]]; then
   echo "The 2nd argument must be mute/unmute"
   exit 1
fi

process_id=$(pidof "$1")

if [ $? -ne 0 ]; then
   echo "There is no such process as "$1""
   exit 1
fi

temp=$(mktemp)

pacmd list-sink-inputs > $temp

inputs_found=0;
current_index=-1;

while read line; do
   if [ $inputs_found -eq 0 ]; then
      inputs=$(echo -ne "$line" | awk '{print $2}')
      if [[ "$inputs" == "to" ]]; then
         continue
      fi
      inputs_found=1
   else
      if [[ "${line:0:6}" == "index:" ]]; then
         current_index="${line:7}"
      elif [[ "${line:0:25}" == "application.process.id = " ]]; then
         if [[ "${line:25}" == "\"$process_id\"" ]]; then
            #index found...
            break;
         fi
      fi
   fi
done < $temp

rm -f $temp

if [ $current_index -eq -1 ]; then
   echo "Could not find "$1" in the processes that output sound."
   exit 1
fi

#muting...
if [[ "$2" == "mute" ]]; then
   pacmd set-sink-input-mute "$current_index" 1 > /dev/null 2>&1
else
   pacmd set-sink-input-mute "$current_index" 0 > /dev/null 2>&1
fi

exit 0

Você pode trabalhar com é como:

./mute_application.sh spotify mute

ou

./mute_application.sh spotify unmute

Testado com o Audacious e o Vlc executando e desativando / desativando apenas um deles.

hytromo
fonte
Script perfeito, funciona como esperado
era878 26/08/12
1

Realmente não sei escrever, mas modifiquei o script do hakermania para criar outro.

Este aumentará ou diminuirá o volume do aplicativo específico em incrementos de 5%:

editar: na verdade, ele está funcionando para alterar sempre o último aplicativo aberto. Ideias?

#!/bin/bash
# Script to increase or decrease an individual application's volume using PulseAudio, depending solely on
# process name, based on another script by hakermania, constructed as answer on askubuntu.com: 
# http://askubuntu.com/questions/180612/script-to-mute-an-application

# It works as: change_app_volume.sh vlc increase OR change_app_volume.sh vlc decrease
# Set desired increments in lines #66 and #68

if [ -z "$1" ]; then
   echo "Please provide me with an application name"
   exit 1
fi

if [ -z "$2" ]; then
   echo "Please provide me with an action increase/decrease after the application name"
   exit 1
fi

if ! [[ "$2" == "increase" || "$2" == "decrease" ]]; then
   echo "The 2nd argument must be increase/decrease"
   exit 1
fi

process_id=$(pidof "$1")

if [ $? -ne 0 ]; then
   echo "There is no such process as "$1""
   exit 1
fi

temp=$(mktemp)

pacmd list-sink-inputs > $temp

inputs_found=0;
current_index=-1;

while read line; do
   if [ $inputs_found -eq 0 ]; then
      inputs=$(echo -ne "$line" | awk '{print $2}')
      if [[ "$inputs" == "to" ]]; then
         continue
      fi
      inputs_found=1
   else
      if [[ "${line:0:6}" == "index:" ]]; then
         current_index="${line:7}"
      elif [[ "${line:0:25}" == "application.process.id = " ]]; then
         if [[ "${line:25}" == "\"$process_id\"" ]]; then
            #index found...
            break;
         fi
      fi
   fi
done < $temp

rm -f $temp

if [ $current_index -eq -1 ]; then
   echo "Could not find "$1" in the processes that output sound."
   exit 1
fi

#increase/decrease...
if [[ "$2" == "increase" ]]; then
   pactl set-sink-input-volume "$current_index" +5% > /dev/null 2>&1
else
   pactl set-sink-input-volume "$current_index" -5% > /dev/null 2>&1
fi

exit 0
RomuNe
fonte
0

Script editado para silenciar todas as entradas de um aplicativo (vários processos) e o padrão é alternar:

#!/bin/bash

main() {
    local action=toggle
    while getopts :hu option; do
        case "$option" in
            h) usage 0 ;;
            m) action=mute ;;
            u) action=unmute ;;
            ?) usage 1 "invalid option: -$OPTARG" ;;
        esac
    done
    shift $((OPTIND - 1))

    if [[ "$1" ]]; then
        $action "$1"
    else
        usage 1 "specify an application name"
    fi
}

usage() {
    [[ "$2" ]] && echo "error: $2"
    echo "usage: $0 [-h] [-u] appname"
    echo "where: -u = ummute , -m = mute (default action is to toggle)"
    exit $1
}

mute()   { adjust_muteness "$1" 1; }
unmute() { adjust_muteness "$1" 0; }
toggle() { adjust_muteness "$1" toggle; }

adjust_muteness() {
    clients=$(pactl list clients short | awk '/[0-9]+.*'$1'.*/{print $1}')
    inputs=$(pactl list sink-inputs short)
    for c in $clients; do
        for i in $(printf '%s' "$inputs" | awk '/[0-9]+\s[0-9]+\s'$c'/{print $1}'); do
            pactl set-sink-input-mute $i $2 &
        done
    done
}

main "$@"
desenterrar
fonte