mirror of
https://github.com/kovidgoyal/kitty
synced 2026-06-08 22:28:24 +02:00
Have the receiving side send progress updates
This is both more accurate and works for rsync based transfers, where we dont know the total size to transmit because of the streaming nature of the rsync protocol.
This commit is contained in:
@@ -5,6 +5,7 @@
|
||||
|
||||
import os
|
||||
import stat
|
||||
from asyncio import TimerHandle
|
||||
from collections import deque
|
||||
from enum import auto
|
||||
from itertools import count
|
||||
@@ -81,7 +82,8 @@ class File:
|
||||
self.err_msg = ''
|
||||
self.actual_file: Optional[IO[bytes]] = None
|
||||
self.transmitted_bytes = 0
|
||||
self.transmit_started_at = self.transmit_ended_at = 0.
|
||||
self.reported_progress = 0
|
||||
self.transmit_started_at = self.transmit_ended_at = self.done_at = 0.
|
||||
self.signature_loader: Optional[LoadSignature] = None
|
||||
self.delta_loader: Optional[Iterator[memoryview]] = None
|
||||
|
||||
@@ -250,6 +252,7 @@ class ProgressTracker:
|
||||
self.transfered_stats_amt = 0
|
||||
self.transfered_stats_interval = 0.
|
||||
self.started_at = 0.
|
||||
self.total_reported_progress = 0
|
||||
|
||||
def change_active_file(self, nf: File) -> None:
|
||||
now = monotonic()
|
||||
@@ -260,7 +263,7 @@ class ProgressTracker:
|
||||
self.transfers.append(Transfer())
|
||||
self.started_at = monotonic()
|
||||
|
||||
def on_transfer(self, amt: int) -> None:
|
||||
def on_transmit(self, amt: int) -> None:
|
||||
if self.active_file is not None:
|
||||
self.active_file.transmitted_bytes += amt
|
||||
self.total_transferred += amt
|
||||
@@ -271,12 +274,19 @@ class ProgressTracker:
|
||||
self.transfered_stats_interval = now - self.transfers[0].at
|
||||
self.transfered_stats_amt = sum(t.amt for t in self.transfers)
|
||||
|
||||
def on_file_progress(self, af: File, delta: int) -> None:
|
||||
self.total_reported_progress += delta
|
||||
|
||||
def on_file_done(self, af: File) -> None:
|
||||
af.done_at = monotonic()
|
||||
|
||||
|
||||
class SendManager:
|
||||
|
||||
def __init__(
|
||||
self, request_id: str, files: Tuple[File, ...],
|
||||
bypass: Optional[str] = None, use_rsync: bool = False,
|
||||
file_progress: Callable[[File, int], None] = lambda f, i: None,
|
||||
file_done: Callable[[File], None] = lambda f: None,
|
||||
):
|
||||
self.use_rsync = use_rsync
|
||||
@@ -293,6 +303,7 @@ class SendManager:
|
||||
self.suffix = '\x1b\\'
|
||||
self.progress = ProgressTracker(sum(df.file_size for df in self.files if df.file_size >= 0))
|
||||
self.file_done = file_done
|
||||
self.file_progress = file_progress
|
||||
|
||||
@property
|
||||
def active_file(self) -> Optional[File]:
|
||||
@@ -305,7 +316,6 @@ class SendManager:
|
||||
if self.active_idx is not None:
|
||||
paf = self.files[self.active_idx]
|
||||
paf.transmit_ended_at = monotonic()
|
||||
self.file_done(paf)
|
||||
for i, f in enumerate(self.files):
|
||||
if f.state is FileState.transmitting:
|
||||
self.active_idx = i
|
||||
@@ -365,16 +375,28 @@ class SendManager:
|
||||
file.state = FileState.waiting_for_data if ftc.ttype is TransmissionType.rsync else FileState.transmitting
|
||||
if file.state is FileState.waiting_for_data:
|
||||
file.signature_loader = LoadSignature()
|
||||
self.update_collective_statuses()
|
||||
elif ftc.status == 'PROGRESS':
|
||||
change = ftc.size - file.reported_progress
|
||||
file.reported_progress = ftc.size
|
||||
self.progress.on_file_progress(file, change)
|
||||
self.file_progress(file, change)
|
||||
else:
|
||||
if ftc.name and not file.remote_final_path:
|
||||
file.remote_final_path = ftc.name
|
||||
file.state = FileState.acknowledged
|
||||
if ftc.status != 'OK':
|
||||
if ftc.status == 'OK':
|
||||
change = ftc.size - file.reported_progress
|
||||
file.reported_progress = ftc.size
|
||||
self.progress.on_file_progress(file, change)
|
||||
self.file_progress(file, change)
|
||||
else:
|
||||
file.err_msg = ftc.status
|
||||
self.progress.on_file_done(file)
|
||||
self.file_done(file)
|
||||
if self.active_idx is not None and file is self.files[self.active_idx]:
|
||||
self.file_done(self.files[self.active_idx])
|
||||
self.active_idx = None
|
||||
self.update_collective_statuses()
|
||||
self.update_collective_statuses()
|
||||
|
||||
def on_signature_data_received(self, ftc: FileTransmissionCommand) -> None:
|
||||
file = self.fid_map.get(ftc.file_id)
|
||||
@@ -404,7 +426,8 @@ class Send(Handler):
|
||||
|
||||
def __init__(self, cli_opts: TransferCLIOptions, files: Tuple[File, ...]):
|
||||
Handler.__init__(self)
|
||||
self.manager = SendManager(random_id(), files, cli_opts.permissions_bypass, cli_opts.transmit_deltas, self.on_file_done)
|
||||
self.manager = SendManager(
|
||||
random_id(), files, cli_opts.permissions_bypass, cli_opts.transmit_deltas, self.on_file_progress, self.on_file_done)
|
||||
self.cli_opts = cli_opts
|
||||
self.transmit_started = False
|
||||
self.file_metadata_sent = False
|
||||
@@ -417,6 +440,7 @@ class Send(Handler):
|
||||
self.done_files: List[File] = []
|
||||
self.failed_files: List[File] = []
|
||||
self.transmit_ok_checked = False
|
||||
self.progress_update_call: Optional[TimerHandle] = None
|
||||
|
||||
def send_payload(self, payload: str) -> None:
|
||||
self.write(self.manager.prefix)
|
||||
@@ -527,7 +551,7 @@ class Send(Handler):
|
||||
def on_writing_finished(self) -> None:
|
||||
chunk_transmitted = self.manager.current_chunk_uncompressed_sz is not None
|
||||
if chunk_transmitted:
|
||||
self.manager.progress.on_transfer(self.manager.current_chunk_uncompressed_sz or 0)
|
||||
self.manager.progress.on_transmit(self.manager.current_chunk_uncompressed_sz or 0)
|
||||
self.manager.current_chunk_uncompressed_sz = None
|
||||
if self.quit_after_write_code is not None:
|
||||
self.quit_loop(self.quit_after_write_code)
|
||||
@@ -602,11 +626,21 @@ class Send(Handler):
|
||||
self.cmd.clear_to_end_of_screen()
|
||||
self.progress_drawn = False
|
||||
|
||||
def schedule_progress_update(self, delay: float = 0.1) -> None:
|
||||
if self.progress_update_call is None:
|
||||
self.progress_update_call = self.asyncio_loop.call_later(delay, self.refresh_progress)
|
||||
elif self.asyncio_loop.time() + delay < self.progress_update_call.when():
|
||||
self.progress_update_call.cancel()
|
||||
self.progress_update_call = self.asyncio_loop.call_later(delay, self.refresh_progress)
|
||||
|
||||
def on_file_progress(self, file: File, change: int) -> None:
|
||||
self.schedule_progress_update()
|
||||
|
||||
def on_file_done(self, file: File) -> None:
|
||||
self.done_files.append(file)
|
||||
if file.err_msg:
|
||||
self.failed_files.append(file)
|
||||
self.asyncio_loop.call_soon(self.refresh_progress)
|
||||
self.schedule_progress_update()
|
||||
|
||||
@Handler.atomic_update
|
||||
def draw_progress(self) -> None:
|
||||
@@ -631,12 +665,12 @@ class Send(Handler):
|
||||
self.print()
|
||||
self.render_progress(
|
||||
'Total', spinner_char=sc,
|
||||
bytes_so_far=p.total_transferred, total_bytes=p.total_bytes_to_transfer,
|
||||
bytes_so_far=p.total_reported_progress, total_bytes=p.total_bytes_to_transfer,
|
||||
secs_so_far=now - p.started_at, is_complete=is_complete,
|
||||
bytes_per_sec=safe_divide(p.transfered_stats_amt, p.transfered_stats_interval)
|
||||
)
|
||||
self.print()
|
||||
self.asyncio_loop.call_later(self.spinner.interval, self.refresh_progress)
|
||||
self.schedule_progress_update(self.spinner.interval)
|
||||
self.progress_drawn = True
|
||||
|
||||
def refresh_progress(self) -> None:
|
||||
@@ -648,8 +682,8 @@ class Send(Handler):
|
||||
now = monotonic()
|
||||
self.render_progress(
|
||||
af.display_name, spinner_char=spinner_char, is_complete=is_complete,
|
||||
bytes_so_far=af.transmitted_bytes, total_bytes=af.bytes_to_transmit,
|
||||
secs_so_far=(af.transmit_ended_at or now) - af.transmit_started_at,
|
||||
bytes_so_far=af.reported_progress, total_bytes=af.bytes_to_transmit,
|
||||
secs_so_far=(af.done_at or now) - af.transmit_started_at,
|
||||
bytes_per_sec=safe_divide(p.transfered_stats_amt, p.transfered_stats_interval)
|
||||
)
|
||||
|
||||
|
||||
Reference in New Issue
Block a user