import os, sys

os.chdir(sys.path[0])
sys.path.append("../")
os.getcwd()

from os import system
from typing import List

from paramiko import AutoAddPolicy, RSAKey, SSHClient
from paramiko.auth_handler import AuthenticationException, SSHException
from scp import SCPClient, SCPException

# from scp_http.log import LOGGER
import logging


class RemoteClient:
    """Client to interact with a remote host via SSH & SCP."""

    def __init__(
        self,
        host: str,
        user: str,
        password: str,
        ssh_key_filepath: str,
    ):
        self.host = host
        self.user = user
        self.password = password
        self.ssh_key_filepath = ssh_key_filepath
        self.client = None
        self._upload_ssh_key()

    @property
    def connection(self):
        """Open SSH connection to remote host."""
        try:
            client = SSHClient()
            client.load_system_host_keys()
            client.set_missing_host_key_policy(AutoAddPolicy())
            client.connect(
                self.host,
                username=self.user,
                password=self.password,
                key_filename=self.ssh_key_filepath,
                timeout=5000,
            )
            return client
        except AuthenticationException as e:
            logging.error(f"AuthenticationException occurred; did you remember to generate an SSH key? {e}")
        except Exception as e:
            logging.error(f"Unexpected error occurred while connecting to host: {e}")

    @property
    def scp(self) -> SCPClient:
        def progress(filename, size, sent):
            sys.stdout.write("%s's progress: %.2f%%   \r" % (filename, float(sent) / float(size) * 100))

        def progress4(filename, size, sent, peername):
            sys.stdout.write(
                "(%s:%s) %s's progress: %.2f%%   \r"
                % (peername[0], peername[1], filename, float(sent) / float(size) * 100)
            )

        conn = self.connection
        return SCPClient(conn.get_transport())

    def _get_ssh_key(self):
        """Fetch locally stored SSH key."""
        try:
            self.ssh_key = RSAKey.from_private_key_file(self.ssh_key_filepath)
            logging.info(f"Found SSH key at self {self.ssh_key_filepath}")
            return self.ssh_key
        except SSHException as e:
            logging.error(f"SSHException while getting SSH key: {e}")
        except Exception as e:
            logging.error(f"Unexpected error while getting SSH key: {e}")

    def _upload_ssh_key(self):
        try:
            system(f"ssh-copy-id -i {self.ssh_key_filepath}.pub {self.user}@{self.host}>/dev/null 2>&1")
            logging.info(f"{self.ssh_key_filepath} uploaded to {self.host}")
        except FileNotFoundError as e:
            logging.error(f"FileNotFoundError while uploading SSH key: {e}")
        except Exception as e:
            logging.error(f"Unexpected error while uploading SSH key: {e}")

    def progress(filename, size, sent):
        sys.stdout.write("%s's progress: %.2f%%   \r" % (filename, float(sent) / float(size) * 100))

    def progress4(filename, size, sent, peername):
        sys.stdout.write(
            "(%s:%s) %s's progress: %.2f%%   \r" % (peername[0], peername[1], filename, float(sent) / float(size) * 100)
        )

    def disconnect(self):
        """Close SSH & SCP connection."""
        if self.connection:
            self.connection.close()
        if self.scp:
            self.scp.close()

    def bulk_upload_together(self, filepaths: List[str]):
        """
        Upload multiple files to a remote directory.

        :param List[str] filepaths: List of local files to be uploaded.
        """
        try:
            self.scp.put(filepaths, remote_path=self.remote_path, recursive=True)
            logging.info(f"Finished uploading {len(filepaths)} files to {self.remote_path} on {self.host}")

        except SCPException as e:
            logging.error(f"SCPException during bulk upload: {e}")
        except Exception as e:
            logging.error(f"Unexpected exception during bulk upload: {e}")

    def bulk_upload(self, filepaths: List[str], logfile=None):
        """
        Upload multiple files to a remote directory.

        :param List[str] filepaths: List of local files to be uploaded.
        """
        for filepath in filepaths:
            existing_files = []
            if logfile != None:
                with open(logfile, "r") as f:
                    existing_files = f.readlines()
                    existing_files = [x.strip() for x in existing_files]
            if filepath not in existing_files:
                try:
                    self.scp.put(filepath, remote_path=self.remote_path, recursive=True)
                    logging.info(f"Finished uploading {filepath} to {self.remote_path} on {self.host}")
                    with open(logfile, "a+") as f:
                        f.write(filepath + "\n")

                except SCPException as e:
                    logging.error(f"SCPException during bulk upload: {e}")
                except Exception as e:
                    logging.error(f"Unexpected exception during bulk upload: {e}")

    def bulk_upload_one(self, filepath, remotepath):
        try:
            self.scp.put(filepath, remote_path=remotepath, recursive=True)
            print(
                f">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   Finished uploading {filepath} to {remotepath} on {self.host}   <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<"
            )

        except SCPException as e:
            logging.error(f"SCPException during bulk upload: {e}")
        except Exception as e:
            logging.error(f"Unexpected exception during bulk upload: {e}")

    def download_file(self, filepath: str):
        """
        Download file from remote host.

        :param str filepath: Path to file hosted on remote server to fetch.
        """
        self.scp.get(filepath)

    def execute_commands(self, commands: List[str]):
        """
        Execute multiple commands in succession.

        :param List[str] commands: List of unix commands as strings.
        """
        for cmd in commands:
            stdin, stdout, stderr = self.connection.exec_command(cmd)
            stdout.channel.recv_exit_status()
            response = stdout.readlines()
            for line in response:
                logging.info(
                    f"INPUT: {cmd}\n \
                    OUTPUT: {line}"
                )
