1
0
mirror of https://codeberg.org/privacy1st/subprocess-util synced 2025-01-11 00:36:05 +01:00

print transfer stats

This commit is contained in:
Daniel Langbein 2023-01-11 20:02:36 +01:00
parent 0109fd27c7
commit 097c195e0e
3 changed files with 73 additions and 4 deletions

@ -56,10 +56,10 @@ def parse_args():
) )
parser.add_argument('--chunk-size', parser.add_argument('--chunk-size',
help='Size in bytes; defaults to 64 MB.', help='Size in bytes; defaults to 128 MB.',
dest='chunk_size', dest='chunk_size',
type=int, type=int,
default=64 * 1024 * 1024, default=128 * 1024 * 1024,
) )
parser.add_argument('child', parser.add_argument('child',

50
src/p1st/data_units.py Normal file

@ -0,0 +1,50 @@
from enum import Enum
def test():
for num_bytes in [0, 1, 128, 512, 1024, 4096, 75 * 1000, 750 * 1000, 850 * 1000, 1111 * 1000]:
print(f'{num_bytes} bytes = {DataUnitConverter.to_unit_auto_str(num_bytes)}')
class DataUnitConverter:
class DataUnit(Enum):
B = 'B'
KB = 'KB'
MB = 'MB'
GB = 'GB'
TB = 'TB'
# Units sorted from small to large.
_unit_dict: dict[DataUnit, int] = {
DataUnit.B: 1,
DataUnit.KB: 1024,
DataUnit.MB: 1024 * 1024,
DataUnit.GB: 1024 * 1024 * 1024,
DataUnit.TB: 1024 * 1024 * 1024 * 1024,
}
@classmethod
def to_bytes(cls, num: int, unit: DataUnit) -> int:
return num * cls._unit_dict[unit]
@classmethod
def to_unit(cls, num_bytes: int, unit: DataUnit = DataUnit.GB) -> float:
return round(num_bytes / cls._unit_dict[unit], 3)
@classmethod
def to_unit_auto(cls, num_bytes: int) -> (float, DataUnit):
for unit, factor in cls._unit_dict.items():
converted = cls.to_unit(num_bytes, unit)
if converted <= 999:
return converted, unit
return num_bytes, cls.DataUnit.B
@classmethod
def to_unit_auto_str(cls, num_bytes: int) -> str:
value, unit = cls.to_unit_auto(num_bytes)
return f'{value} {unit.value}'
if __name__ == '__main__':
test()

@ -1,5 +1,7 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
import datetime
import time
from pathlib import Path from pathlib import Path
import sys import sys
import threading import threading
@ -7,18 +9,20 @@ import subprocess
from typing import AnyStr, IO, Callable from typing import AnyStr, IO, Callable
from p1st.common import _get_chunk_file from p1st.common import _get_chunk_file
from p1st.data_units import DataUnitConverter
def _rotate_chunk(chunk_file: Path, chunk_number: int, eof: bool, chunk_transfer_fun: Callable, chunk_transfer_args: tuple): def _rotate_chunk(chunk_file: Path, chunk_number: int, eof: bool, chunk_transfer_fun: Callable, chunk_transfer_args: tuple):
print(f'Transferring chunk {chunk_file}') print(f'Transferring chunk')
chunk_transfer_fun(chunk_file, chunk_number, eof, *chunk_transfer_args) chunk_transfer_fun(chunk_file, chunk_number, eof, *chunk_transfer_args)
print(f'Removing chunk {chunk_file}') print(f'Removing chunk')
chunk_file.unlink(missing_ok=False) chunk_file.unlink(missing_ok=False)
def _save_chunk(chunk: bytes, chunk_file: Path): def _save_chunk(chunk: bytes, chunk_file: Path):
print(f'Saving chunk {chunk_file}') print(f'Saving chunk {chunk_file}')
# Fails if file does already exist. # Fails if file does already exist.
with open(chunk_file, 'xb') as f: with open(chunk_file, 'xb') as f:
f.write(chunk) f.write(chunk)
@ -44,6 +48,8 @@ def _save_output_rotating_chunks(out_pipe: IO[AnyStr],
chunk_transfer_args: tuple, chunk_transfer_args: tuple,
get_chunk_file: Callable[[Path, int], Path], get_chunk_file: Callable[[Path, int], Path],
): ):
start_time = time.time()
ct: int = 1 ct: int = 1
remaining_bytes = chunk_size remaining_bytes = chunk_size
chunk: bytes = b'' chunk: bytes = b''
@ -60,6 +66,11 @@ def _save_output_rotating_chunks(out_pipe: IO[AnyStr],
_save_chunk(chunk, chunk_file) _save_chunk(chunk, chunk_file)
_rotate_chunk(chunk_file, ct, True, chunk_transfer_fun, chunk_transfer_args) _rotate_chunk(chunk_file, ct, True, chunk_transfer_fun, chunk_transfer_args)
current_time = time.time()
elapsed_time = current_time - start_time
print(f'Elapsed time: {datetime.timedelta(seconds=elapsed_time)}\n'
f'Transferred: {DataUnitConverter.to_unit_auto_str(ct * chunk_size)}')
break break
chunk += b chunk += b
@ -72,6 +83,14 @@ def _save_output_rotating_chunks(out_pipe: IO[AnyStr],
_save_chunk(chunk, chunk_file) _save_chunk(chunk, chunk_file)
_rotate_chunk(chunk_file, ct, False, chunk_transfer_fun, chunk_transfer_args) _rotate_chunk(chunk_file, ct, False, chunk_transfer_fun, chunk_transfer_args)
current_time = time.time()
elapsed_time = current_time - start_time
transferred_bytes = ct * chunk_size
bytes_per_second = transferred_bytes / elapsed_time
print(f'Elapsed time: {datetime.timedelta(seconds=elapsed_time)}\n'
f'Transferred: {DataUnitConverter.to_unit_auto_str(transferred_bytes)}\n'
f'Speed: {DataUnitConverter.to_unit_auto_str(bytes_per_second)}/s')
chunk = b'' chunk = b''
remaining_bytes = chunk_size remaining_bytes = chunk_size
ct += 1 ct += 1