import requests, tarfile, io, os, zipfile, gzip
from typing import Optional

from io import BytesIO, SEEK_SET, SEEK_END
from tqdm import tqdm


class UrlStream:
    def __init__(self, url, headers={}):
        self._url = url
        self._head = headers
        rq = requests.head(url, headers={"Accept-Encoding": "identity", **self._head})
        if rq.status_code == 302:
            self._url = rq.headers['location']
            rq = requests.head(self._url, headers={"Accept-Encoding": "identity", **self._head})

        headers = {k.lower(): v for k, v in rq.headers.items()}
        self._seek_supported = headers.get('accept-ranges') == 'bytes' and 'content-length' in headers
        if self._seek_supported:
            self._size = int(headers['content-length'])
        self._curr_pos = 0
        self._buf_start_pos = 0
        self._iter = None
        self._buffer = None
        self._buf_size = 0
        self._loaded_all = False

    def _load_all(self):
        if self._loaded_all:
            return
        self._make_request()
        old_buf_pos = self._buffer.tell()
        self._buffer.seek(0, SEEK_END)
        for chunk in self._iter:
            self._buffer.write(chunk)
        self._buf_size = self._buffer.tell()
        self._buffer.seek(old_buf_pos, SEEK_SET)
        self._loaded_all = True

    def seekable(self):
        return self._seek_supported

    def seek(self, position, whence=SEEK_SET):
        if whence == SEEK_END:
            assert position <= 0
            if self._seek_supported:
                self.seek(self._size + position)
            else:
                self._load_all()
                self._buffer.seek(position, SEEK_END)
                self._curr_pos = self._buffer.tell()
        elif whence == SEEK_SET:
            if self._curr_pos != position:
                self._curr_pos = position
                if self._seek_supported:
                    self._iter = None
                    self._buffer = None
                else:
                    self._load_until(position)
                    self._buffer.seek(position)
                    self._curr_pos = position
        else:
            assert "Invalid whence %s" % whence

        return self.tell()

    def tell(self):
        return self._curr_pos

    def _load_until(self, goal_position):
        retries = 0

        while True:
            try:
                self._make_request()
                old_buf_pos = self._buffer.tell()
                self._buffer.seek(0, SEEK_END)

                goal_position = goal_position - self._buf_start_pos
                while self._buf_size < goal_position:
                    try:
                        d = next(self._iter)
                        self._buffer.write(d)
                        self._buf_size += len(d)
                    except StopIteration:
                        break
                self._buffer.seek(old_buf_pos, SEEK_SET)
            except requests.exceptions.ChunkedEncodingError as e:
                # This happens if the server is overloaded, in the next() call from above. Restore buffer, reset the
                # connection and try again. Unfortunately we can't hook into requests to partial data downloaded,
                # and also we can't use the data in it's internal buffer.

                if retries >= 5:
                    print(f"Multiple consecutive failures trying to download a chink from {self._url}. Giving up.")
                    raise

                retries += 1
                self._buffer.seek(old_buf_pos, SEEK_SET)
                self._iter = None
            else:
                break

    def _new_buffer(self):
        remaining = self._buffer.read() if self._buffer is not None else None
        self._buffer = BytesIO()
        if remaining is not None:
            self._buffer.write(remaining)
        self._buf_start_pos = self._curr_pos
        self._buf_size = 0 if remaining is None else len(remaining)
        self._buffer.seek(0, SEEK_SET)
        self._loaded_all = False

    def _make_request(self):
        if self._iter is None:
            h = {
                "User-agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/47.0.2526.80 Safari/537.36",
                **self._head
            }
            if self._seek_supported:
                h["Range"] = "bytes=%d-%d" % (self._curr_pos + (self._buf_size if self._buffer is not None else 0), self._size - 1)

            r = requests.get(self._url, headers=h, stream=True)

            self._iter = r.iter_content(1024 * 1024)
            self._new_buffer()
        elif self._seek_supported and self._buf_size > 128 * 1024 * 1024:
            self._new_buffer()

    def size(self):
        if self._seek_supported:
            return self._size
        else:
            self._load_all()
            return self._buf_size

    def read(self, size=None):
        if size is None:
            size = self.size()

        self._load_until(self._curr_pos + size)
        if self._seek_supported:
            self._curr_pos = min(self._curr_pos + size, self._size)

        read_data = self._buffer.read(size)
        if not self._seek_supported:
            self._curr_pos += len(read_data)
        return read_data

    def iter_content(self, block_size):
        while True:
            d = self.read(block_size)
            if not len(d):
                break
            yield d


def download(url: str, dest: Optional[str] = None, extract: bool=True, ignore_if_exists: bool = False,
             compression: Optional[str] = None, headers={}):
    """
    Download a file from the internet.

    Args:
        url: the url to download
        dest: destination file if extract=False, or destionation dir if extract=True. If None, it will be the last part of URL.
        extract: extract a tar.gz or zip file?
        ignore_if_exists: don't do anything if file exists

    Returns:
        the destination filename.
    """

    base_url = url.split("?")[0]
    url_end = [f for f in base_url.split("/") if f][-1]

    if dest is None:
        dest = url_end

    stream = UrlStream(url, headers=headers)
    extension = base_url.split(".")[-1].lower()

    if extract and extension in ['gz', 'bz2', 'zip', 'tgz', 'tar']:
        if os.path.exists(dest) and ignore_if_exists:
            return dest

        os.makedirs(dest, exist_ok=True)

        if extension == "gz" and not base_url.endswith(".tar.gz"):
            decompressed_file = gzip.GzipFile(fileobj=stream)
            with open(os.path.join(dest, url.split("/")[-1][:-3]), 'wb') as f:
                while True:
                    d = decompressed_file.read(1024 * 1024)
                    if not d:
                        break
                    f.write(d)
        else:
            if extension in ['gz', 'bz2', "tgz", "tar"]:
                decompressed_file = tarfile.open(fileobj=stream, mode='r|' +
                                                                      (compression or (
                                                                          "gz" if extension == "tgz" else extension)))
            elif extension == 'zip':
                decompressed_file = zipfile.ZipFile(stream, mode='r')
            else:
                assert False, "Invalid extension: %s" % extension

            decompressed_file.extractall(dest)
    else:
        if dest.endswith("/"):
            dest = os.path.join(dest, url_end)

        if os.path.exists(dest) and ignore_if_exists:
            return dest

        try:
            p = tqdm(total=stream.size())
            with open(dest, 'wb') as f:
                for d in stream.iter_content(1024 * 1024):
                    f.write(d)
                    p.update(len(d))

        except:
            os.remove(dest)
            raise
    return dest