60 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			60 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Python
		
	
	
	
from __future__ import annotations
 | 
						|
 | 
						|
import io
 | 
						|
from typing import Callable
 | 
						|
from typing_extensions import override
 | 
						|
 | 
						|
 | 
						|
class CancelledError(Exception):
 | 
						|
    def __init__(self, msg: str) -> None:
 | 
						|
        self.msg = msg
 | 
						|
        super().__init__(msg)
 | 
						|
 | 
						|
    @override
 | 
						|
    def __str__(self) -> str:
 | 
						|
        return self.msg
 | 
						|
 | 
						|
    __repr__ = __str__
 | 
						|
 | 
						|
 | 
						|
class BufferReader(io.BytesIO):
 | 
						|
    def __init__(self, buf: bytes = b"", desc: str | None = None) -> None:
 | 
						|
        super().__init__(buf)
 | 
						|
        self._len = len(buf)
 | 
						|
        self._progress = 0
 | 
						|
        self._callback = progress(len(buf), desc=desc)
 | 
						|
 | 
						|
    def __len__(self) -> int:
 | 
						|
        return self._len
 | 
						|
 | 
						|
    @override
 | 
						|
    def read(self, n: int | None = -1) -> bytes:
 | 
						|
        chunk = io.BytesIO.read(self, n)
 | 
						|
        self._progress += len(chunk)
 | 
						|
 | 
						|
        try:
 | 
						|
            self._callback(self._progress)
 | 
						|
        except Exception as e:  # catches exception from the callback
 | 
						|
            raise CancelledError("The upload was cancelled: {}".format(e)) from e
 | 
						|
 | 
						|
        return chunk
 | 
						|
 | 
						|
 | 
						|
def progress(total: float, desc: str | None) -> Callable[[float], None]:
 | 
						|
    import tqdm
 | 
						|
 | 
						|
    meter = tqdm.tqdm(total=total, unit_scale=True, desc=desc)
 | 
						|
 | 
						|
    def incr(progress: float) -> None:
 | 
						|
        meter.n = progress
 | 
						|
        if progress == total:
 | 
						|
            meter.close()
 | 
						|
        else:
 | 
						|
            meter.refresh()
 | 
						|
 | 
						|
    return incr
 | 
						|
 | 
						|
 | 
						|
def MB(i: int) -> int:
 | 
						|
    return int(i // 1024**2)
 |