blob: 265b3c180030f843fdb65334c4c98b4d8a6374f6 [file] [log] [blame] [edit]
#!/bin/bash
# Copyright 2020 The Fuchsia Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
#### CATEGORY=Other
### Remotely run emulator
## usage: fx emu-remote [--no-build] [--stream [--no-emu] [--no-turn] [--no-open] [--display DPY] [--port PORT]] HOST [DIR] [-- ARGS]
##
## Connect to HOST, run a build using fx from DIR, fetch the artifacts and
## start the emulator. Alternatively, start the emulator on HOST,
## and open an WebRTC connection to it using local browser.
##
## --no-build do not build, just pull artifacts already present
## --stream stream output from remote emulator using WebRTC instead of fetching artifacts
## --no-emu only tunnel, do not start remote emulator
## --no-turn do not use turn configuration for remote emulator
## --no-open do not open https://web-femu.appspot.com, just run remote emulator
## --display DPY do not start remote virtual display, use DPY instead
## --port PORT port used on local machine to connect with remote emulator over HTTP (default: 8080)
##
## HOST the hostname to connect to
## DIR defaults to ~/fuchsia, the path to the FUCHSIA_DIR on HOST
## ARGS arguments to pass to emulator
##
## HOST and DIR are persisted in the file //.fx-remote-config and are reused as
## defaults in future invocations of any 'fx *-remote' tools.
source "$(cd "$(dirname "${BASH_SOURCE[0]}")" >/dev/null 2>&1 && pwd)"/lib/vars.sh || exit $?
source "$(cd "$(dirname "${BASH_SOURCE[0]}")" >/dev/null 2>&1 && pwd)"/lib/fx-remote.sh || exit $?
fx-config-read
build=true
stream=false
emu=true
turn=true
open=true
display="xvfb-run"
host=""
dir=""
local_port="8080"
remote_port="8080"
while [[ $# -ne 0 ]]; do
case "$1" in
--help|-h)
fx-command-help
exit 0
;;
--no-build)
build=false
;;
--stream)
stream=true
;;
--no-emu)
emu=false
;;
--no-turn)
turn=false
;;
--no-open)
open=false
;;
--display)
shift
display="DISPLAY=$1"
;;
--port)
shift
local_port=$1
;;
--)
shift
break
;;
*)
if [[ -z "${host}" ]]; then
host="$1"
elif [[ -z "${dir}" ]]; then
dir="$1"
else
fx-error "unexpected argument: '$1'"
exit 1
fi
;;
esac
shift
done
if cached=( $(load_remote_info "$host") ); then
host="${cached[0]}"
dir="${cached[1]}"
fi
if [[ -z "${host}" ]]; then
fx-error "HOST must be specified"
fx-command-help
exit 1
fi
# Use a dedicated ControlPath so script can manage a connection separately from the user's. We
# intentionally do not use %h/%p in the control path because there can only be one forwarding
# session at a time (due to the local forward of 8083).
ssh_base_args=(
"${host}"
-S "~/.ssh/control-fuchsia-fx-remote"
-o ControlMaster=auto
)
ssh_exit() {
ssh "${ssh_base_args[@]}" -O exit > /dev/null 2>&1
wait # for ssh to exit
}
# If there is already control master then exit it. We can't be sure its to the right host and it
# also could be stale.
ssh_exit
# When we exit the script, close the background ssh connection.
trap_exit() {
ssh_exit
exit
}
trap trap_exit EXIT
if [[ -z "${dir}" ]]; then
echo ssh "${ssh_base_args[@]}" ls \~/fuchsia/.jiri_root/bin/fx
if ssh "${ssh_base_args[@]}" ls \~/fuchsia/.jiri_root/bin/fx > /dev/null; then
dir="~/fuchsia"
else
fx-error "failed to find ~/fuchsia on $host, please specify DIR"
fx-command-help
exit 1
fi
fi
emu_targets=(multiboot.bin fuchsia.zbi obj/build/images/fvm.blk)
if "${build}"; then
ssh "${ssh_base_args[@]}" "cd ${dir} && ./.jiri_root/bin/fx build ${emu_targets[@]}" || exit $?
fi
build_dir=$(ssh "${ssh_base_args[@]}" "cd ${dir} && ./.jiri_root/bin/fx get-build-dir")
save_remote_info "$host" "$dir"
# Fetch artifacts and run local emulator when streaming is disabled.
if [[ "${stream}" == "false" ]]; then
build_arch=$(ssh "${ssh_base_args[@]}" "cd ${dir} && ./.jiri_root/bin/fx exec sh -c 'echo \${FUCHSIA_ARCH}'")
mkdir -p out/fetched
printf -v files '%s,' "${emu_targets[@]}"
rsync -z -P "${host}":"${build_dir}/{${files%,}}" out/fetched/
echo >&2 "Build images fetched into out/fetched"
fx-command-exec emu \
-K out/fetched/multiboot.bin \
-z out/fetched/fuchsia.zbi \
-f out/fetched/fvm.blk \
-A "${build_arch}" "$@"
fi
#
# The code below handles the case where output from a remote emulator
# is streamed using WebRTC to a local browser instance.
#
# Check if the port given in argument "$1" is listened by other processes.
#
# Returns true if other process is already listening to this port; otherwise
# returns false.
is_port_used() {
# Print all listening ports and check if $1 is listed.
if [[ "$(uname -s)" == "Darwin" ]]; then
# On macOS we parse output from `netstat`.
netstat -anp tcp 2>/dev/null |
awk '$1 ~ /tcp4/ && $6 == "LISTEN" { \
sub(/.*\./, "", $4); print $4 \
}'
else
# On Linux we parse output from `ss`.
ss -lpn4 2>/dev/null | awk '$1 == "tcp" { \
if (match($5, /^[0-9.]+:[0-9]+$/)) { \
sub(/^.*:/, "", $5); print $5 \
} \
}'
fi | grep -q "^$1$"
}
if is_port_used ${local_port}; then
fx-error "Local port ${local_port} is not available. Please try specifying"\
"a different port using --port flag."
exit 1
fi
echo "Using remote ${host}:${dir}"
wait_and_open() {
timeout=10
waited=0
# Loop until connection is live or timeout expires.
while true
do
# Check if we can connect to the forwarded port.
if curl -sI "http://localhost:${local_port}" > /dev/null; then
open_args=("https://web-femu.appspot.com/?port=${local_port}")
# Launch Chrome browser tab for emulator feed.
if [[ "$(uname -s)" == "Darwin" ]]; then
exec open -a "/Applications/Google Chrome.app" "${open_args[@]}"
else
exec google-chrome "${open_args[@]}"
fi
fi
# Continue waiting if we have not reached our timeout.
if [ $waited -gt 0 ]; then
if [ $waited -ge $timeout ]; then
echo "No emulator after waiting $waited seconds, giving up."
break
fi
echo "Waiting for emulator ($(($timeout-$waited)) seconds left).."
fi
sleep 1
waited=$(($waited+1))
done
}
ssh_args=(
-6 # We want ipv6 binds for the port forwards.
# Requests to the WebRTC service address locally go to the workstation.
-L "${local_port}:localhost:${remote_port}"
-o ExitOnForwardFailure=yes
)
if "${emu}"; then
# If the user requested emulator, then we'll check to see if there's a
# emulator already running and kill it, this prevents most cases where
# signal propagation not make it to "qemu-system".
if ssh "${ssh_base_args[@]}" 'pgrep -u $USER goldfish-webrtc' > /dev/null; then
ssh "${ssh_base_args[@]}" 'pkill -u $USER qemu-system; pkill -u $USER goldfish-webrtc'
fi
emu_args=(-x ${remote_port})
# Add turn configuration if enabled.
if "${turn}"; then
apikey="AIzaSyBl6TgfdN6FKAQ3nK2GvKNcKNjWLeRGVYQ"
emu_args+=(-t "\"curl -s -X POST https://networktraversal.googleapis.com/v1alpha/iceconfig?key=${apikey}\"")
fi
# Starts emulator.
ssh_args+=("cd ${dir} && ${display} ./.jiri_root/bin/fx emu ${emu_args[@]} $@")
# Wait and open page for emulator when ready.
if "${open}"; then
wait_and_open &
fi
else
# Starts nothing, just goes to sleep.
ssh_args+=("-nNT")
fi
# Cleanup after ssh command ends:
# - Terminate all child processes;
# - Close port forwarding if we use ssh multiplexing.
cleanup_processes_and_ports() {
child_pids=$(jobs -p)
if [[ -n "${child_pids}" ]]; then
# Note: child_pids must be expanded to args here.
kill ${child_pids} 2> /dev/null
wait 2> /dev/null
fi
# Cancel port forwarding.
if ssh -O check "${host}" > /dev/null 2>&1; then
ssh -O cancel -L "${local_port}:localhost:${remote_port}" "${host}"
fi
}
trap cleanup_processes_and_ports EXIT
ssh "${ssh_base_args[@]}" "${ssh_args[@]}"