В Ubuntu приложения можно открывать из терминала. Но иногда неясно, какая команда должна это сделать.
Итак, имея открытое приложение, как я могу использовать команду для его запуска, без необходимости искать в любом месте (просто глядя на него)?
Я только что сделал следующий скрипт, который использует заголовок окна приложения, чтобы узнать правильную команду, открывающую соответствующее приложение из терминала (я назвал его appcmd
):
#!/bin/bash
#appcmd - script which use the application window title to find out the right command which opens the respective application from terminal
#Licensed under the standard MIT license:
#Copyright 2013 Radu Rădeanu (http://askubuntu.com/users/147044/).
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE
#check if wmctrl is installed
if [ ! -n "$(dpkg -s wmctrl 2>/dev/null | grep 'Status: install ok installed')" ]; then
echo -e "The package 'wmctrl' must to be installed before to run $(basename $0).\nUse 'sudo apt-get install wmctrl' command to install it."
exit
fi
window_title=$(echo $@ | awk '{print tolower($0)}')
windows=$(mktemp)
pids=$(mktemp)
pid_found=""
wmctrl -l | awk '{$2=$3=""; print $0}' > $windows
cat $windows | while read identity window; do
if [[ $(echo $window | awk '{print tolower($0)}') == *$window_title* ]]; then
wmctrl -lp | grep -e "$identity.*$window" | awk '{$1=$2=$4=""; print $0}'
fi
done > $pids
while read pid window; do
if [ "$pid" != "0" -a "$window" != "Desktop" ]; then
echo -e "Application window title:\t$window"
echo -e "Command to open from terminal:\t\$ $(ps -o command $pid | tail -n 1)\n"
pid_found="$pid"
fi
done < $pids
if [ "$pid_found" = "" ]; then
echo "There is no any opened application containing '$@' in the window title."
fi
Сохраните этот скрипт в вашем каталоге ~/bin
и не забудьте сделать его исполняемым:
chmod +x ~/bin/appcmd
Usage:
Когда скрипт выполняется без каких-либо аргументов, скрипт вернет все соответствующие команды для всех открытых окон.
Если задан какой-либо аргумент, скрипт попытается найти открытое окно приложения, содержащее в своем заголовке этот аргумент, и вернет соответствующую команду. Например, если открыт браузер Chromium, то команду, открывающую его из терминала, можно узнать только при помощи:
appcmd chromium
Из here:
xprop | awk '($1=="_NET_WM_PID(CARDINAL)") {print $3}' | xargs ps h -o pid,cmd
Если вам нужна только стартовая командная строка, то просто:
xprop | awk '($1=="_NET_WM_PID(CARDINAL)") {print $3}' | xargs ps h -o cmd
После того, как вы запустили команду, просто щелкните по окну, для которого вы хотите, чтобы была отображена стартовая команда.
Другой способ перечислить имя команды и аргументы запущенных процессов:
ps axk pid,comm o comm,args > output.txt
(Перенаправить в файл, чтобы имена / аргументы команд не усекались.)
Источник: man ps
: раздел примеров (с небольшими изменениями).
Системный монитор - это графический интерфейс для ps
.
Альтернативный сценарий:
#!/bin/bash
# Copyright © 2013 minerz029
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
shopt -s extglob
for var in 'wm_pid' 'wm_name' 'wm_class' 'cmdline' 'wm_id'; do
declare "$var"'=Not found'
done
notify-send -t 3000 'Click on a window to get the command line...'
xprop_out="$(xprop)"
while IFS=$'\n' read -r -d $'\n' line; do
if [[ "$line" == '_NET_WM_PID(CARDINAL) = '* ]]; then
wm_pid="${line#_NET_WM_PID(CARDINAL) = }"
elif [[ "$line" == 'WM_NAME('?(UTF8_)'STRING) = '* ]]; then
wm_name="${line#WM_NAME(?(UTF8_)STRING) = }"
elif [[ "$line" == 'WM_CLASS('?(UTF8_)'STRING) = '* ]]; then
wm_class="${line#WM_CLASS(?(UTF8_)STRING) = }"
elif [[ "$line" == 'WM_CLIENT_LEADER(WINDOW): window id # '* ]]; then
wm_id="${line#WM_CLIENT_LEADER(WINDOW): window id # }"
fi
done <<< "$xprop_out"
if [[ "$wm_pid" == +([0-9]) ]]; then
quote ()
{
local quoted="${1//\'/\'\\\'\'}";
out="$(printf "'%s'" "$quoted")"
if eval echo -n "$out" >/dev/null 2>&1; then
echo "$out"
else
echo "SEVERE QUOTING ERROR"
echo "IN: $1"
echo -n "OUT: "
eval echo -n "$out"
fi
}
cmdline=()
while IFS= read -d '' -r arg; do
cmdline+=("$(quote "$arg")")
done < "/proc/$wm_pid/cmdline"
fi
text="\
Title:
$wm_name
Class:
$wm_class
ID:
$wm_id
PID:
$wm_pid
Command line:
${cmdline[@]}"
copy() {
{ echo -n "$1" | xsel -i -b >/dev/null; } && xsel -k
}
if [[ -t 1 ]]; then
echo "$text"
if [[ "$1" == '--copy' ]]; then
echo "Copied"
copy "$cmdline"
fi
else
zenity \
--title='Window information' \
--width=750 \
--height=300 \
--no-wrap \
--font='Ubuntu Mono 11' \
--text-info \
--cancel-label='Copy' \
--ok-label='Close' \
<<< "$text"
if [[ $? == 1 ]]; then
copy "$cmdline"
fi
fi
Использование:
В качестве альтернативы без использования сценария вы можете просто открыть системный монитор и навести указатель мыши на процесс которую вы хотели бы узнать о командной строке.
Если вы включите «Просмотр зависимостей», вы сможете увидеть, какой процесс вызвал другой, например, вы можете увидеть различные процессы, которые Chrome создает для каждой вкладки, и отследить их до родительского процесса, который будет иметь командная строка, с которой был запущен Chrome (пользователем).
Самая похожая мысль, которую я нашёл, это xwinfo, которая даёт вам информацию о запущенном окне. Но она не сообщает вам, какая программа запущена внутри него.