#!/bin/sh # Copyright (C) 2022 Kovid Goyal # Distributed under terms of the GPLv3 license. # read the transmitted data from STDIN cleanup_on_bootstrap_exit() { [ -n "$saved_tty_settings" ] && command stty "$saved_tty_settings" 2> /dev/null < /dev/tty saved_tty_settings="" } die() { printf "\033[31m%s\033[m\n\r" "$*" > /dev/stderr; cleanup_on_bootstrap_exit; exit 1; } dcs_to_kitty() { printf "\033P@kitty-$1|%s\033\\" "$(printf "%s" "$2" | base64 | tr -d \\n)" > /dev/tty; } debug() { dcs_to_kitty "print" "debug $1"; } echo_via_kitty() { dcs_to_kitty "echo" "$1"; } saved_tty_settings=$(command stty -g 2> /dev/null < /dev/tty) tty_ok="n" [ -n "$saved_tty_settings" ] && tty_ok="y" if [ "$tty_ok" = "y" ]; then command stty raw min 1 time 0 -echo 2> /dev/null < /dev/tty || die "stty failed to set raw mode" trap 'cleanup_on_bootstrap_exit' EXIT fi data_started="n" data_complete="n" if [ -z "$HOSTNAME" ]; then hostname=$(hostname 2> /dev/null) if [ -z "$hostname" ]; then hostname=$(hostnamectl hostname 2> /dev/null) if [ -z "$hostname" ]; then hostname="_"; fi fi else hostname="$HOSTNAME" fi # ensure $HOME is set if [ -z "$HOME" ]; then HOME=~; fi # ensure $USER is set if [ -z "$USER" ]; then USER=$(whoami 2> /dev/null); fi # ask for the SSH data data_password="DATA_PASSWORD" password_filename="PASSWORD_FILENAME" data_complete="n" leading_data="" [ "$tty_ok" = "y" ] && dcs_to_kitty "ssh" "hostname="$hostname":pwfile="$password_filename":user="$USER":pw="$data_password"" record_separator=$(printf "\036") mv_files_and_dirs() { cwd="$PWD"; cd "$1"; command find . -type d -exec mkdir -p "$2/{}" ";" command find . -type l -exec sh -c "tgt=\$(command readlink -n \"{}\"); command ln -sf \"\$tgt\" \"$2/{}\"; command rm -f \"{}\"" ";" command find . -type f -exec mv "{}" "$2/{}" ";" cd "$cwd"; } untar_and_read_env() { # extract the tar file atomically, in the sense that any file from the # tarfile is only put into place after it has been fully written to disk tdir=$(mktemp -d "$HOME/.kitty-ssh-kitten-untar-XXXXXXXXXXXX"); [ $? = 0 ] || die "Creating temp directory failed"; # using dd with bs=1 is very slow, so use head. On non GNU coreutils head # does not limit itself to reading -c bytes only from the pipe so we can potentially lose # some trailing data, for instance if the user starts typing. Cant be helped. command head -c "$1" < /dev/tty | command base64 -d | command tar xjf - --no-same-owner -C "$tdir"; data_file="$tdir/data.sh"; [ -f "$data_file" ] && . "$data_file"; data_dir="$HOME/$KITTY_SSH_KITTEN_DATA_DIR" mv_files_and_dirs "$tdir/home" "$HOME" [ -e "$tdir/root" ] && mv_files_and_dirs "$tdir/root" "" command rm -rf "$tdir"; [ -z "KITTY_SSH_KITTEN_DATA_DIR" ] && die "Failed to read SSH data from tty"; unset KITTY_SSH_KITTEN_DATA_DIR; } read_record() { # We need a way to read a single byte at a time and to read a specified number of bytes in one invocation. # The options are head -c, read -N and dd # # read -N is not in POSIX and dash/posh dont implement it. Also bash seems to read beyond # the specified number of bytes into an internal buffer. # # head -c reads beyond the specified number of bytes into an internal buffer on macOS # # POSIX dd works for one byte at a time but for reading X bytes it needs the GNU iflag=count_bytes # extension, and is anyway unsafe as it can lead to corrupt output when the read syscall is interrupted. record="" while :; do n=$(command dd bs=1 count=1 2> /dev/null < /dev/tty) [ "$n" = "$record_separator" ] && break record="$record$n" done printf "%s" "$record" } get_data() { leading_data=$(read_record) size=$(read_record) case "$size" in ("!"*) die "$size" ;; esac untar_and_read_env "$size" } if [ "$tty_ok" = "y" ]; then get_data command stty "$saved_tty_settings" 2> /dev/null saved_tty_settings="" if [ -n "$leading_data" ]; then # clear current line as it might have things echoed on it from leading_data # because we only turn off echo in this script whereas the leading bytes could # have been sent before the script had a chance to run printf "\r\033[K" fi shell_integration_dir="$data_dir/shell-integration" [ -f "$HOME/.terminfo/kitty.terminfo" ] || die "Incomplete extraction of ssh data"; # export TERMINFO tname=".terminfo" if [ -e "/usr/share/misc/terminfo.cdb" ]; then # NetBSD requires this see https://github.com/kovidgoyal/kitty/issues/4622 tname=".terminfo.cdb" fi export TERMINFO="$HOME/$tname" # compile terminfo for this system if [ -x "$(command -v tic)" ]; then tic_out=$(command tic -x -o "$HOME/$tname" "$HOME/.terminfo/kitty.terminfo" 2>&1) rc=$? if [ "$rc" != "0" ]; then die "$tic_out"; fi fi fi login_shell_is_ok() { if [ -z "$login_shell" -o ! -x "$login_shell" ]; then return 1; fi case "$login_shell" in *sh) return 0; esac return 1; } detect_python() { python=$(command -v python3) if [ -z "$python" ]; then python=$(command -v python2); fi if [ -z "$python" ]; then python=$(command -v python); fi if [ -z "$python" -o ! -x "$python" ]; then return 1; fi return 0; } parse_passwd_record() { printf "%s" "$(grep -o '[^:]*$')" } using_getent() { cmd=$(command -v getent) if [ -n "$cmd" ]; then output=$($cmd passwd $USER 2>/dev/null) if [ $? = 0 ]; then login_shell=$(echo $output | parse_passwd_record); if login_shell_is_ok; then return 0; fi fi fi return 1; } using_id() { cmd=$(command -v id) if [ -n "$cmd" ]; then output=$($cmd -P $USER 2>/dev/null) if [ $? = 0 ]; then login_shell=$(echo $output | parse_passwd_record); if login_shell_is_ok; then return 0; fi fi fi return 1; } using_passwd() { if [ -f "/etc/passwd" -a -r "/etc/passwd" ]; then output=$(grep "^$USER:" /etc/passwd 2>/dev/null) if [ $? = 0 ]; then login_shell=$(echo $output | parse_passwd_record); if login_shell_is_ok; then return 0; fi fi fi return 1; } using_python() { if detect_python; then output=$($python -c "import pwd, os; print(pwd.getpwuid(os.geteuid()).pw_shell)") if [ $? = 0 ]; then login_shell=$output; if login_shell_is_ok; then return 0; fi fi fi return 1 } execute_with_python() { if detect_python; then exec $python -c "import os; os.execl('$login_shell', '-' '$shell_name')" fi return 1; } LOGIN_SHELL="OVERRIDE_LOGIN_SHELL" if [ -n "$LOGIN_SHELL" ]; then login_shell="$LOGIN_SHELL" else using_getent || using_id || using_python || using_passwd || die "Could not detect login shell"; fi shell_name=$(command basename $login_shell) # If a command was passed to SSH execute it here EXEC_CMD if [ "$tty_ok" = "n" ]; then if [ -z "$(command -v stty)" ]; then printf "%s\n" "stty missing ssh kitten cannot function" > /dev/stderr else printf "%s\n" "stty failed ssh kitten cannot function" > /dev/stderr fi fi exec_bash_with_integration() { export ENV="$shell_integration_dir/bash/kitty.bash" export KITTY_BASH_INJECT="1" if [ -z "$HISTFILE" ]; then export HISTFILE="$HOME/.bash_history" export KITTY_BASH_UNEXPORT_HISTFILE="1" fi exec "$login_shell" "--posix" } exec_zsh_with_integration() { zdotdir="$ZDOTDIR" if [ -z "$zdotdir" ]; then zdotdir=~; unset KITTY_ORIG_ZDOTDIR # ensure this is not propagated else export KITTY_ORIG_ZDOTDIR="$zdotdir" fi # dont prevent zsh-new-user from running if [ -f "$zdotdir/.zshrc" -o -f "$zdotdir/.zshenv" -o -f "$zdotdir/.zprofile" -o -f "$zdotdir/.zlogin" ]; then export ZDOTDIR="$shell_integration_dir/zsh" exec "$login_shell" "-l" fi } exec_fish_with_integration() { if [ -z "$XDG_DATA_DIRS" ]; then export XDG_DATA_DIRS="$shell_integration_dir" else export XDG_DATA_DIRS="$shell_integration_dir:$XDG_DATA_DIRS" fi export KITTY_FISH_XDG_DATA_DIR="$shell_integration_dir" exec "$login_shell" "-l" } exec_with_shell_integration() { case "$shell_name" in "zsh") exec_zsh_with_integration ;; "bash") exec_bash_with_integration ;; "fish") exec_fish_with_integration ;; esac } case "$KITTY_SHELL_INTEGRATION" in ("") # only blanks or unset unset KITTY_SHELL_INTEGRATION ;; (*) # not blank q=$(printf "%s" "$KITTY_SHELL_INTEGRATION" | grep '\bno-rc\b') if [ -z "$q" ]; then exec_with_shell_integration # exec failed, unset unset KITTY_SHELL_INTEGRATION fi ;; esac # We need to pass the first argument to the executed program with a leading - # to make sure the shell executes as a login shell. Note that not all shells # support exec -a so we use the below to try to detect such shells if [ -z "$PIPESTATUS" ]; then # the dash shell does not support exec -a and also does not define PIPESTATUS execute_with_python exec $login_shell "-l" fi exec -a "-$shell_name" $login_shell