500 lines
13 KiB
Bash
500 lines
13 KiB
Bash
# /* vim: set ai ts=4 ft=sh: */
|
|
#
|
|
# Copyright 2011, The Android Open Source Project
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
#
|
|
|
|
_adb() {
|
|
if ! check_type "$1" >/dev/null; then
|
|
return
|
|
fi
|
|
|
|
if check_type _init_completion >/dev/null; then
|
|
_init_completion || return
|
|
fi
|
|
|
|
local where i cur serial
|
|
COMPREPLY=()
|
|
|
|
serial="${ANDROID_SERIAL:-none}"
|
|
where=OPTIONS
|
|
for ((i=1; i <= COMP_CWORD; i++)); do
|
|
cur="${COMP_WORDS[i]}"
|
|
case "${cur}" in
|
|
-s)
|
|
where=OPT_SERIAL
|
|
;;
|
|
-p)
|
|
where=OPT_PATH
|
|
;;
|
|
-*)
|
|
where=OPTIONS
|
|
;;
|
|
*)
|
|
if [[ $where == OPT_SERIAL ]]; then
|
|
where=OPT_SERIAL_ARG
|
|
serial=${cur}
|
|
else
|
|
where=COMMAND
|
|
break
|
|
fi
|
|
;;
|
|
esac
|
|
done
|
|
|
|
if [[ $where == COMMAND && $i -ge $COMP_CWORD ]]; then
|
|
where=OPTIONS
|
|
fi
|
|
|
|
OPTIONS="-d -e -s -p"
|
|
COMMAND="devices connect disconnect push pull sync shell emu logcat lolcat forward jdwp install uninstall bugreport help version start-server kill-server get-state get-serialno status-window remount reboot reboot-bootloader root usb tcpip disable-verity"
|
|
|
|
case $where in
|
|
OPTIONS|OPT_SERIAL|OPT_PATH)
|
|
COMPREPLY=( $(compgen -W "$OPTIONS $COMMAND" -- "$cur") )
|
|
;;
|
|
OPT_SERIAL_ARG)
|
|
local devices=$(command adb devices 2> /dev/null | grep -v "List of devices" | awk '{ print $1 }')
|
|
COMPREPLY=( $(compgen -W "${devices}" -- ${cur}) )
|
|
;;
|
|
COMMAND)
|
|
if [[ $i -eq $COMP_CWORD ]]; then
|
|
COMPREPLY=( $(compgen -W "$COMMAND" -- "$cur") )
|
|
else
|
|
i=$((i+1))
|
|
case "${cur}" in
|
|
install)
|
|
_adb_cmd_install "$serial" $i
|
|
;;
|
|
sideload)
|
|
_adb_cmd_sideload "$serial" $i
|
|
;;
|
|
pull)
|
|
_adb_cmd_pull "$serial" $i
|
|
;;
|
|
push)
|
|
_adb_cmd_push "$serial" $i
|
|
;;
|
|
reboot)
|
|
if [[ $COMP_CWORD == $i ]]; then
|
|
args="bootloader recovery"
|
|
COMPREPLY=( $(compgen -W "${args}" -- "${COMP_WORDS[i]}") )
|
|
fi
|
|
;;
|
|
shell)
|
|
_adb_cmd_shell "$serial" $i
|
|
;;
|
|
uninstall)
|
|
_adb_cmd_uninstall "$serial" $i
|
|
;;
|
|
esac
|
|
fi
|
|
;;
|
|
esac
|
|
|
|
return 0
|
|
}
|
|
|
|
_adb_cmd_install() {
|
|
local serial i cur where
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
where=OPTIONS
|
|
for ((; i <= COMP_CWORD; i++)); do
|
|
cur="${COMP_WORDS[i]}"
|
|
case "${cur}" in
|
|
-*)
|
|
where=OPTIONS
|
|
;;
|
|
*)
|
|
where=FILE
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
if [[ $where == OPTIONS ]]; then
|
|
COMPREPLY=( $(compgen -W "-d -l -r -s" -- "${cur}") )
|
|
return
|
|
fi
|
|
|
|
_adb_util_complete_local_file "${cur}" '!*.apk'
|
|
}
|
|
|
|
_adb_cmd_sideload() {
|
|
local serial i cur
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
|
|
_adb_util_complete_local_file "${cur}" '!*.zip'
|
|
}
|
|
|
|
_adb_cmd_push() {
|
|
local serial IFS=$'\n' i cur
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
|
|
if [[ $COMP_CWORD == $i ]]; then
|
|
_adb_util_complete_local_file "${cur}"
|
|
elif [[ $COMP_CWORD == $(($i+1)) ]]; then
|
|
if [ "${cur}" == "" ]; then
|
|
cur="/"
|
|
fi
|
|
_adb_util_list_files $serial "${cur}"
|
|
fi
|
|
}
|
|
|
|
_adb_cmd_pull() {
|
|
local serial IFS=$'\n' i cur
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
|
|
if [[ $COMP_CWORD == $i ]]; then
|
|
if [ "${cur}" == "" ]; then
|
|
cur="/"
|
|
fi
|
|
_adb_util_list_files $serial "${cur}"
|
|
elif [[ $COMP_CWORD == $(($i+1)) ]]; then
|
|
_adb_util_complete_local_file "${cur}"
|
|
fi
|
|
}
|
|
|
|
_adb_cmd_shell() {
|
|
local serial IFS=$'\n' i cur
|
|
local -a args
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
cur="${COMP_WORDS[i]}"
|
|
if [ "$serial" != "none" ]; then
|
|
args=(-s $serial)
|
|
fi
|
|
|
|
if [[ $i -eq $COMP_CWORD && ${cur:0:1} != "/" ]]; then
|
|
paths=$(command adb ${args[@]} shell echo '$'PATH 2> /dev/null | tr -d '\r' | tr : '\n')
|
|
COMMAND=$(command adb ${args[@]} shell ls $paths '2>' /dev/null | tr -d '\r' | {
|
|
while read -r tmp; do
|
|
command=${tmp##*/}
|
|
printf '%s\n' "$command"
|
|
done
|
|
})
|
|
COMPREPLY=( $(compgen -W "$COMMAND" -- "$cur") )
|
|
return 0
|
|
fi
|
|
|
|
i=$((i+1))
|
|
case "$cur" in
|
|
ls)
|
|
_adb_shell_file_command $serial $i "--color -A -C -F -H -L -R -S -Z -a -c -d -f -h -i -k -l -m -n -p -q -r -s -t -u -x -1"
|
|
;;
|
|
cat)
|
|
_adb_shell_file_command $serial $i "-h -e -t -u -v"
|
|
;;
|
|
dumpsys)
|
|
_adb_cmd_shell_dumpsys "$serial" $i
|
|
;;
|
|
am)
|
|
_adb_cmd_shell_am "$serial" $i
|
|
;;
|
|
pm)
|
|
_adb_cmd_shell_pm "$serial" $i
|
|
;;
|
|
/*)
|
|
_adb_util_list_files $serial "$cur"
|
|
;;
|
|
*)
|
|
COMPREPLY=( )
|
|
;;
|
|
esac
|
|
|
|
return 0
|
|
}
|
|
|
|
_adb_cmd_shell_dumpsys() {
|
|
local serial i cur
|
|
local -a args
|
|
local candidates
|
|
|
|
unset IFS
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
if [ "$serial" != "none" ]; then
|
|
args=(-s $serial)
|
|
fi
|
|
|
|
if (( $i == $COMP_CWORD )) ; then
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
# First line is a header, so need "1d".
|
|
candidates=$(command adb ${args[@]} shell dumpsys -l 2> /dev/null | sed -e '1d;s/^ *//' | tr -d '\r')
|
|
candidates="-l $candidates"
|
|
COMPREPLY=( $(compgen -W "$candidates" -- "$cur") )
|
|
return 0
|
|
fi
|
|
|
|
COMPREPLY=( )
|
|
return 0
|
|
}
|
|
|
|
_adb_cmd_shell_am() {
|
|
local serial i cur
|
|
local candidates
|
|
|
|
unset IFS
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
if (( $i == $COMP_CWORD )) ; then
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
candidates="broadcast clear-debug-app clear-watch-heap dumpheap force-stop get-config get-inactive hang idle-maintenance instrument kill kill-all monitor package-importance profile restart screen-compat send-trim-memory set-debug-app set-inactive set-watch-heap stack start startservice start-user stopservice stop-user suppress-resize-config-changes switch-user task to-app-uri to-intent-uri to-uri"
|
|
COMPREPLY=( $(compgen -W "$candidates" -- "$cur") )
|
|
return 0
|
|
fi
|
|
|
|
COMPREPLY=( )
|
|
return 0
|
|
}
|
|
|
|
|
|
_adb_cmd_shell_pm() {
|
|
local serial i cur
|
|
local candidates
|
|
|
|
unset IFS
|
|
|
|
serial=$1
|
|
i=$2
|
|
|
|
if (( $i == $COMP_CWORD )) ; then
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
candidates="-l -lf -p clear create-user default-state disable"
|
|
candidates+=" disable-until-used disable-user dump enable"
|
|
candidates+=" get-app-link get-install-location get-max-users"
|
|
candidates+=" get-max-running-users grant hide install"
|
|
candidates+=" install-abandon install-commit install-create"
|
|
candidates+=" install-write list move-package"
|
|
candidates+=" move-primary-storage path remove-user"
|
|
candidates+=" reset-permissions revoke set-app-link"
|
|
candidates+=" set-installer set-install-location"
|
|
candidates+=" set-permission-enforced trim-caches unhide"
|
|
candidates+=" uninstall"
|
|
COMPREPLY=( $(compgen -W "$candidates" -- "$cur") )
|
|
return 0
|
|
fi
|
|
|
|
if (( $i + 1 == $COMP_CWORD )) && [[ "${COMP_WORDS[COMP_CWORD -1]}" == "list" ]] ; then
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
candidates="packages permission-groups permissions instrumentation features libraries users"
|
|
COMPREPLY=( $(compgen -W "$candidates" -- "$cur") )
|
|
return 0
|
|
fi
|
|
|
|
COMPREPLY=( )
|
|
return 0
|
|
}
|
|
|
|
_adb_cmd_uninstall() {
|
|
local serial i where cur packages
|
|
|
|
serial=$1
|
|
i=$2
|
|
if [ "$serial" != "none" ]; then
|
|
args=(-s $serial)
|
|
fi
|
|
|
|
where=OPTIONS
|
|
for ((; i <= COMP_CWORD; i++)); do
|
|
cur="${COMP_WORDS[i]}"
|
|
case "${cur}" in
|
|
-*)
|
|
where=OPTIONS
|
|
;;
|
|
*)
|
|
where=FILE
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
|
|
cur="${COMP_WORDS[COMP_CWORD]}"
|
|
if [[ $where == OPTIONS ]]; then
|
|
COMPREPLY=( $(compgen -W "-k" -- "${cur}") )
|
|
fi
|
|
|
|
packages="$(
|
|
command adb ${args[@]} shell pm list packages '2>' /dev/null 2> /dev/null | tr -d '\r' | {
|
|
while read -r tmp; do
|
|
local package=${tmp#package:}
|
|
echo -n "${package} "
|
|
done
|
|
}
|
|
)"
|
|
|
|
COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -W "${packages}" -- "${cur}") )
|
|
}
|
|
|
|
_adb_shell_file_command() {
|
|
local serial i cur file options
|
|
local -a args
|
|
|
|
serial=$1
|
|
i=$2
|
|
if [ "$serial" != "none" ]; then
|
|
args=(-s $serial)
|
|
fi
|
|
options=$3
|
|
|
|
where=OPTIONS
|
|
for ((; i <= COMP_CWORD; i++)); do
|
|
cur="${COMP_WORDS[i]}"
|
|
case "${cur}" in
|
|
-*)
|
|
where=OPTIONS
|
|
;;
|
|
*)
|
|
where=FILE
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
|
|
file="${COMP_WORDS[COMP_CWORD]}"
|
|
if [[ ${file} == "" ]]; then
|
|
file="/"
|
|
fi
|
|
|
|
case $where in
|
|
OPTIONS)
|
|
unset IFS
|
|
COMPREPLY=( $(compgen -W "$options" -- "$cur") )
|
|
;;
|
|
FILE)
|
|
_adb_util_list_files $serial "$file"
|
|
;;
|
|
esac
|
|
|
|
return 0
|
|
}
|
|
|
|
_adb_util_list_files() {
|
|
local serial dir IFS=$'\n'
|
|
local -a toks
|
|
local -a args
|
|
|
|
serial="$1"
|
|
file="$2"
|
|
|
|
if [ "$serial" != "none" ]; then
|
|
args=(-s $serial)
|
|
fi
|
|
|
|
if [[ $( command adb ${args[@]} shell ls -dF / '2>/dev/null' | tr -d '\r' ) == "d /" ]] ; then
|
|
toks=( ${toks[@]-} $(
|
|
command adb ${args[@]} shell ls -dF ${file}"*" '2>' /dev/null 2> /dev/null | tr -d '\r' | {
|
|
while read -r tmp; do
|
|
filetype=${tmp%% *}
|
|
filename=${tmp:${#filetype}+1}
|
|
if [[ ${filetype:${#filetype}-1:1} == d ]]; then
|
|
printf '%s/\n' "$filename"
|
|
else
|
|
printf '%s\n' "$filename"
|
|
fi
|
|
done
|
|
}
|
|
))
|
|
else
|
|
toks=( ${toks[@]-} $(
|
|
command adb ${args[@]} shell ls -dp ${file}"*" '2>/dev/null' 2> /dev/null | tr -d '\r'
|
|
))
|
|
fi
|
|
|
|
# Since we're probably doing file completion here, don't add a space after.
|
|
if [[ $(check_type compopt) == "builtin" ]]; then
|
|
compopt -o nospace
|
|
fi
|
|
|
|
COMPREPLY=( ${COMPREPLY[@]:-} "${toks[@]}" )
|
|
}
|
|
|
|
_adb_util_complete_local_file()
|
|
{
|
|
local file xspec i j IFS=$'\n'
|
|
local -a dirs files
|
|
|
|
file=$1
|
|
xspec=$2
|
|
|
|
# Since we're probably doing file completion here, don't add a space after.
|
|
if [[ $(check_type compopt) == "builtin" ]]; then
|
|
compopt -o plusdirs
|
|
if [[ "${xspec}" == "" ]]; then
|
|
COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -f -- "${cur}") )
|
|
else
|
|
compopt +o filenames
|
|
COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -f -X "${xspec}" -- "${cur}") )
|
|
fi
|
|
else
|
|
# Work-around for shells with no compopt
|
|
|
|
dirs=( $(compgen -d -- "${cur}" ) )
|
|
|
|
if [[ "${xspec}" == "" ]]; then
|
|
files=( ${COMPREPLY[@]:-} $(compgen -f -- "${cur}") )
|
|
else
|
|
files=( ${COMPREPLY[@]:-} $(compgen -f -X "${xspec}" -- "${cur}") )
|
|
fi
|
|
|
|
COMPREPLY=( $(
|
|
for i in "${files[@]}"; do
|
|
local skip=
|
|
for j in "${dirs[@]}"; do
|
|
if [[ $i == $j ]]; then
|
|
skip=1
|
|
break
|
|
fi
|
|
done
|
|
[[ -n $skip ]] || printf "%s\n" "$i"
|
|
done
|
|
))
|
|
|
|
COMPREPLY=( ${COMPREPLY[@]:-} $(
|
|
for i in "${dirs[@]}"; do
|
|
printf "%s/\n" "$i"
|
|
done
|
|
))
|
|
fi
|
|
}
|
|
|
|
|
|
if [[ $(check_type compopt) == "builtin" ]]; then
|
|
complete -F _adb adb
|
|
else
|
|
complete -o nospace -F _adb adb
|
|
fi
|