Сценарій для відключення програми


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?

EDIT: Обидва зі сценаріїв нижче роботи , як очікується, може хто - небудь додати перемикач , який буде перевіряти muted: yesчи , muted: noа потім включити або вимкнути звук відповідно?

EDIT: мені вдалося змінити сценарій glenn jackman, щоб додати перемикач:

#!/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 "$@"

чому б не використовувати 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функцію для деталей.
glenn jackman

3
приємні навички awk :)
hytromo

@glennjackman, Yup, я зрозумів це через деякий час. Я вважаю, що сценарій, який я щойно опублікував, працює коректно зараз.
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>). Коли цей другий вислів збігається, надрукується 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"

Він підтримує трирічне пошукове дерево списку оголошень, щоб ефективно перевірити, чи є в списку поточна назва.

Він також використовує асинхронний API PulseAudio для запитів sink-inputsі set-mute:

pa_context_get_sink_input_info_list()
pa_context_set_sink_input_mute()

Оскільки це просто простий код С, це невелика вага. Перевірте вихідний код та .debпакунок Ubuntu за адресою: індикатор-мутед . Ймовірно, він бив сценарій оболонки на 2-3 порядки.


не працює з версією 1.0.11
Janus Troelsen

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 бігали та глумили / відключали звук лише одного з них.


Ідеальний сценарій, працює як очікувалося
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

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 "$@"
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.