mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-25 21:59:19 +00:00
Implement the progress reporting
This commit is contained in:
parent
6690aa7337
commit
b78c6cc5ed
@ -125,6 +125,7 @@ class BackerUpper:
|
|||||||
upload_directory=self.upload_directory,
|
upload_directory=self.upload_directory,
|
||||||
targets=targets,
|
targets=targets,
|
||||||
start_timestamp=start_timestamp,
|
start_timestamp=start_timestamp,
|
||||||
|
stdout=stdout,
|
||||||
)
|
)
|
||||||
new_backup_dircap = completed.dircap
|
new_backup_dircap = completed.dircap
|
||||||
|
|
||||||
@ -312,8 +313,9 @@ def run_backup(
|
|||||||
upload_directory,
|
upload_directory,
|
||||||
targets,
|
targets,
|
||||||
start_timestamp,
|
start_timestamp,
|
||||||
|
stdout,
|
||||||
):
|
):
|
||||||
progress = BackupProgress(warn, start_timestamp)
|
progress = BackupProgress(warn, start_timestamp, len(targets))
|
||||||
for target in targets:
|
for target in targets:
|
||||||
# Pass in the progress and get back a progress. It would be great if
|
# Pass in the progress and get back a progress. It would be great if
|
||||||
# progress objects were immutable. Then the target's backup would
|
# progress objects were immutable. Then the target's backup would
|
||||||
@ -321,7 +323,7 @@ def run_backup(
|
|||||||
# Currently, BackupProgress is mutable, though, and everything just
|
# Currently, BackupProgress is mutable, though, and everything just
|
||||||
# mutates it.
|
# mutates it.
|
||||||
progress = target.backup(progress, upload_file, upload_directory)
|
progress = target.backup(progress, upload_file, upload_directory)
|
||||||
progress.report()
|
print >>stdout, progress.report(datetime.datetime.now())
|
||||||
return progress.backup_finished()
|
return progress.backup_finished()
|
||||||
|
|
||||||
|
|
||||||
@ -473,9 +475,10 @@ class BackupProgress(object):
|
|||||||
# Would be nice if this data structure were immutable and its methods were
|
# Would be nice if this data structure were immutable and its methods were
|
||||||
# transformations that created a new slightly different object. Not there
|
# transformations that created a new slightly different object. Not there
|
||||||
# yet, though.
|
# yet, though.
|
||||||
def __init__(self, warn, start_timestamp):
|
def __init__(self, warn, start_timestamp, target_count):
|
||||||
self._warn = warn
|
self._warn = warn
|
||||||
self._start_timestamp = start_timestamp
|
self._start_timestamp = start_timestamp
|
||||||
|
self._target_count = target_count
|
||||||
self._files_created = 0
|
self._files_created = 0
|
||||||
self._files_reused = 0
|
self._files_reused = 0
|
||||||
self._files_skipped = 0
|
self._files_skipped = 0
|
||||||
@ -487,10 +490,35 @@ class BackupProgress(object):
|
|||||||
self._compare_contents = {}
|
self._compare_contents = {}
|
||||||
|
|
||||||
|
|
||||||
def report(self):
|
def report(self, now):
|
||||||
pass
|
report_format = (
|
||||||
|
"Backing up {target_progress}/{target_total}... {elapsed} elapsed..."
|
||||||
|
)
|
||||||
|
return report_format.format(
|
||||||
|
target_progress=(
|
||||||
|
self._files_created
|
||||||
|
+ self._files_reused
|
||||||
|
+ self._files_skipped
|
||||||
|
+ self._directories_created
|
||||||
|
+ self._directories_reused
|
||||||
|
+ self._directories_skipped
|
||||||
|
),
|
||||||
|
target_total=self._target_count,
|
||||||
|
elapsed=self._format_elapsed(now - self._start_timestamp),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _format_elapsed(self, elapsed):
|
||||||
|
seconds = elapsed.total_seconds()
|
||||||
|
hours = int(seconds / 3600)
|
||||||
|
minutes = int(seconds / 60 % 60)
|
||||||
|
seconds = int(seconds % 60)
|
||||||
|
return "{}h {}m {}s".format(
|
||||||
|
hours,
|
||||||
|
minutes,
|
||||||
|
seconds,
|
||||||
|
)
|
||||||
|
|
||||||
def backup_finished(self):
|
def backup_finished(self):
|
||||||
end_timestamp = datetime.datetime.now()
|
end_timestamp = datetime.datetime.now()
|
||||||
return BackupComplete(
|
return BackupComplete(
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
|
|
||||||
import os.path
|
import os.path
|
||||||
from cStringIO import StringIO
|
from cStringIO import StringIO
|
||||||
|
from datetime import timedelta
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from twisted.trial import unittest
|
from twisted.trial import unittest
|
||||||
@ -36,6 +37,19 @@ class Backup(GridTestMixin, CLITestMixin, StallMixin, unittest.TestCase):
|
|||||||
mo = re.search(r"(\d)+ files checked, (\d+) directories checked", out)
|
mo = re.search(r"(\d)+ files checked, (\d+) directories checked", out)
|
||||||
return [int(s) for s in mo.groups()]
|
return [int(s) for s in mo.groups()]
|
||||||
|
|
||||||
|
def progress_output(self, out):
|
||||||
|
def parse_timedelta(h, m, s):
|
||||||
|
return timedelta(int(h), int(m), int(s))
|
||||||
|
mos = re.findall(
|
||||||
|
r"Backing up (\d)+/(\d)+\.\.\. (\d+)h (\d+)m (\d+)s elapsed\.\.\.",
|
||||||
|
out,
|
||||||
|
)
|
||||||
|
return list(
|
||||||
|
(int(progress), int(total), parse_timedelta(h, m, s))
|
||||||
|
for (progress, total, h, m, s)
|
||||||
|
in mos
|
||||||
|
)
|
||||||
|
|
||||||
def test_backup(self):
|
def test_backup(self):
|
||||||
self.basedir = "cli/Backup/backup"
|
self.basedir = "cli/Backup/backup"
|
||||||
self.set_up_grid(oneshare=True)
|
self.set_up_grid(oneshare=True)
|
||||||
@ -66,8 +80,6 @@ class Backup(GridTestMixin, CLITestMixin, StallMixin, unittest.TestCase):
|
|||||||
|
|
||||||
d.addCallback(lambda res: do_backup(True))
|
d.addCallback(lambda res: do_backup(True))
|
||||||
def _check0((rc, out, err)):
|
def _check0((rc, out, err)):
|
||||||
print()
|
|
||||||
print(out)
|
|
||||||
self.failUnlessReallyEqual(err, "")
|
self.failUnlessReallyEqual(err, "")
|
||||||
self.failUnlessReallyEqual(rc, 0)
|
self.failUnlessReallyEqual(rc, 0)
|
||||||
(
|
(
|
||||||
@ -92,6 +104,34 @@ class Backup(GridTestMixin, CLITestMixin, StallMixin, unittest.TestCase):
|
|||||||
(files_checked, directories_checked) = self.count_output2(out)
|
(files_checked, directories_checked) = self.count_output2(out)
|
||||||
self.failUnlessReallyEqual(files_checked, 0)
|
self.failUnlessReallyEqual(files_checked, 0)
|
||||||
self.failUnlessReallyEqual(directories_checked, 0)
|
self.failUnlessReallyEqual(directories_checked, 0)
|
||||||
|
|
||||||
|
progress = self.progress_output(out)
|
||||||
|
for left, right in zip(progress[:-1], progress[1:]):
|
||||||
|
# Progress as measured by file count should progress
|
||||||
|
# monotonically.
|
||||||
|
self.assertTrue(
|
||||||
|
left[0] < right[0],
|
||||||
|
"Failed: {} < {}".format(left[0], right[0]),
|
||||||
|
)
|
||||||
|
|
||||||
|
# Total work to do should remain the same.
|
||||||
|
self.assertEqual(left[1], right[1])
|
||||||
|
|
||||||
|
# Amount of elapsed time should only go up. Allow it to
|
||||||
|
# remain the same to account for resolution of the report.
|
||||||
|
self.assertTrue(
|
||||||
|
left[2] <= right[2],
|
||||||
|
"Failed: {} <= {}".format(left[2], right[2]),
|
||||||
|
)
|
||||||
|
|
||||||
|
for element in progress:
|
||||||
|
# Can't have more progress than the total.
|
||||||
|
self.assertTrue(
|
||||||
|
element[0] <= element[1],
|
||||||
|
"Failed: {} <= {}".format(element[0], element[1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
d.addCallback(_check0)
|
d.addCallback(_check0)
|
||||||
|
|
||||||
d.addCallback(lambda res: self.do_cli("ls", "--uri", "tahoe:backups"))
|
d.addCallback(lambda res: self.do_cli("ls", "--uri", "tahoe:backups"))
|
||||||
|
Loading…
x
Reference in New Issue
Block a user