Скрипт для отключения звука приложения

14

Моя цель - отключить приложение Spotify, а не всю систему. С помощью команды: ps -C spotify -o pid=я могу найти идентификатор процесса Spotify, в этом случае идентификатор "22981". С этим процессом ID Я хотел бы найти из этого списка: pacmd list-sink-inputs. Эта команда возвращает список вроде этого:

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"

Теперь я хотел бы сопоставить application.process.id = "22981"входной индекс приемника, который в данном случае является index: 0. Теперь с этим индексом мне нужно было бы выполнить эту команду: pacmd set-sink-input-mute 0 1отключить звук Spotify и pacmd set-sink-input-mute 0 0включить звук Spotify. Для этих команд первое число должно быть заменено найденным ранее индексом, а следующее число является логическим значением для включения или выключения отключения звука. Как я могу поместить это все в сценарий, чтобы я мог получить команду для отключения / включения звука приложения Spotify?

РЕДАКТИРОВАТЬ: Оба приведенных ниже сценария работают должным образом, может кто-нибудь добавить переключатель, который будет проверять muted: yesили muted: noзатем включить или выключить звук соответственно?

РЕДАКТИРОВАТЬ: я был в состоянии изменить сценарий Гленна Джекмана, чтобы добавить переключатель:

#!/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
источник
почему бы не использовать pactl list sink-inputs? тогда он будет работать по сети.
Янус Троелсен

Ответы:

13

Вот мой взгляд на ваш интересный вызов:

#!/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 "$@"
Гленн Джекман
источник
Это также отлично работает
era878
@ era878, мне нравится идея переключения по умолчанию. Однако ваша get_statusфункция найдет только «приглушенные» строки без перекрестной проверки того, что статус принадлежит соответствующему приложению. Перечитайте мою get_indexфункцию для деталей.
Гленн Джекман
3
хорошие навыки awk :)
hytromo
@glennjackman, Ага, я понял это через некоторое время. Я считаю, что скрипт, который я только что опубликовал, теперь работает правильно.
era878
1
Подробности: awk -v var=val. Awk зацикливается на строках 1 на 1, пытается найти соответствие любому из $1 == ...операторов, выполняет код в скобках при сопоставлении и продолжает работу. Первый оператор совпадает со строками, чье первое слово index:, и сохраняет второе слово (SINK INDEX) в idxпеременной. Поэтому idxперезаписывается следующей index: <SINK INDEX>строкой, пока awk не совпадет со вторым оператором ( $1= application.process.id, $2= =, $3= <expected pid val>). Когда этот 2-й оператор совпадает, awk печатает idx(это последняя строка, соответствующая первому оператору index:) и завершается.
KrisWebDev
7

спасибо за решение! Мне удалось использовать предоставленные здесь сценарии, чтобы исправить мою проблему. Поскольку мне пришлось немного их изменить, я присоединяюсь к улучшенной версии.

Причина, по которой оригинальные сценарии не работали для меня, заключается в том, что некоторые приложения могут иметь несколько экземпляров, то есть несколько PID, но, возможно, только один из них производит звук и, таким образом, фактически подключен к Pulseaudio. Поскольку скрипт использует только первый найденный PID, он обычно / не / отключает желаемое приложение.

Итак, вот версия, в которой аргумент - это имя приложения, зарегистрированное в PulseAudio. Вы можете найти это имя, запустив pacmd list-sink-inputsкоманду и ища application.nameполе.

Альтернативным решением было бы отключить / включить звук всех PID с одинаковым именем приложения.

#!/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 "$@"
выщерблять
источник
6

Хотя вопрос касается сценария, я хотел оставить это здесь.

Я написал приложение на C, которое делает это в Ubuntu. Более того, он сидит на панели индикаторов (используя libappindicator) и проверяет, что играет Spotify, через короткие промежутки времени. Если он воспроизводит рекламу (проверяет чёрный список), он отключает Spotify. Если воспроизводится новое объявление, вы просто нажимаете кнопку «Без звука» в меню индикатора, и оно добавляется в черный список.

Что он делает, так это ищет окно X, для которого XFetchNameвозвращается Spotify - Linux Preview. Затем он вызывает XGetWindowPropertyзапрос _NET_WM_ICON_NAMEсвойства этого окна, которое возвращает строку в "Spotify – <Artist> – <Song>"формате. При воспроизведении рекламы она возвращает что-то вроде этого:

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

Он поддерживает троичное дерево поиска в списке объявлений, чтобы эффективно проверить, есть ли текущий заголовок в списке.

Он также использует PulseAudio Asynchronous API для запроса sink-inputsи set-mute:

pa_context_get_sink_input_info_list()
pa_context_set_sink_input_mute()

Так как это просто C-код, он легкий. Проверьте исходный код и .debпакет Ubuntu по адресу: Indicator-Muteads . Вероятно, он побьет сценарий оболочки на 2-3 порядка.

mkayaalp
источник
не работает с версией 1.0.11
Янус Троелсен
4

Прежде всего, «более правильный» способ найти PID приложения, такой как spotify, заключается в использовании:

pidof spotify

Я создал скрипт, который выполняет эту работу, я не знаю, является ли это лучшим способом сделать это, но он работает отлично:

#!/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

Вы можете работать как:

./mute_application.sh spotify mute

или

./mute_application.sh spotify unmute

Протестировано с работающими Audacious и Vlc и отключением / включением только одного из них.

hytromo
источник
Идеальный сценарий, работает как положено
era878
1

Я действительно не могу писать сценарии, но я изменил сценарий Хакермании, чтобы создать другой.

Этот будет увеличивать или уменьшать объем конкретного приложения с шагом 5%:

редактировать: на самом деле, он работает всегда меняя последнее открытое приложение. Идеи?

#!/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
источник
0

Отредактированный скрипт для отключения всех входов приложения (несколько процессов) и по умолчанию переключение:

#!/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 "$@"
untore
источник