Sat, 20 Jan 2018 23:19:16 +0000
Display original/deduplicatd size if no progress percentage available
|
1
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
1 | # |
|
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
2 | # MacOS UI |
|
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
3 | # |
|
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
4 | |
|
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
5 | import rumps |
| 10 | 6 | import time |
| 7 | import datetime | |
| 8 | import logging | |
| 9 | from threading import Lock | |
| 10 | ||
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
11 | INACTIVE=0 |
|
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
12 | ACTIVE=1 |
|
16
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
13 | OFFLINE=2 |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
14 | ERRORS=3 |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
15 | |
|
16
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
16 | traynames={ |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
17 | INACTIVE: 'B.', |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
18 | ACTIVE: 'B!', |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
19 | OFFLINE: 'B⦙', |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
20 | ERRORS: 'B?' |
|
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
21 | } |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
22 | |
|
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
23 | def combine_state(state1, state2): |
|
16
d0ffae5550ef
Added offline symbol B⦙ (no offline detection yet)
Tuomo Valkonen <tuomov@iki.fi>
parents:
15
diff
changeset
|
24 | return max(state1, state2) |
|
11
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
25 | |
|
19
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
26 | # Based on code snatched from |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
27 | # https://stackoverflow.com/questions/12523586/python-format-size-application-converting-b-to-kb-mb-gb-tb/37423778 |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
28 | def humanbytes(B): |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
29 | 'Return the given bytes as a human friendly KB, MB, GB, or TB string' |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
30 | B = float(B) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
31 | KB = float(1024) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
32 | MB = float(KB ** 2) # 1,048,576 |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
33 | GB = float(KB ** 3) # 1,073,741,824 |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
34 | TB = float(KB ** 4) # 1,099,511,627,776 |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
35 | |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
36 | if B < KB: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
37 | return '{0} {1}B'.format(B) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
38 | elif KB <= B < MB: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
39 | return '{0:.2f}KB'.format(B/KB) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
40 | elif MB <= B < GB: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
41 | return '{0:.2f}MB'.format(B/MB) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
42 | elif GB <= B < TB: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
43 | return '{0:.2f}GB'.format(B/GB) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
44 | elif TB <= B: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
45 | return '{0:.2f}TB'.format(B/TB) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
46 | |
| 10 | 47 | def make_title(status): |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
48 | state=INACTIVE |
| 15 | 49 | |
| 10 | 50 | if status['type']=='scheduled': |
| 51 | # Operation scheduled | |
| 52 | when=status['when'] | |
| 53 | now=time.time() | |
| 54 | if when<now: | |
| 55 | whenstr='overdue' | |
| 56 | else: | |
| 57 | diff=datetime.timedelta(seconds=when-now) | |
| 58 | if diff.days>0: | |
| 59 | whenday=datetime.date.fromtimestamp(when) | |
| 60 | whenstr='on %s' % whenday.isoformat() | |
| 61 | else: | |
| 62 | twhen=time.localtime(when) | |
| 63 | if twhen.tm_sec>30: | |
| 64 | # Round up minute display to avoid user confusion | |
| 65 | twhen=time.localtime(when+30) | |
| 66 | whenstr='at %02d:%02d' % (twhen.tm_hour, twhen.tm_min) | |
| 67 | detail='' | |
| 68 | if 'detail' in status and status['detail']: | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
69 | if status['detail']=='retry': |
| 15 | 70 | state=ERRORS |
| 10 | 71 | detail=status['detail']+' ' |
|
11
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
72 | title="%s (%s%s %s)" % (status['name'], detail, status['operation'], whenstr) |
| 10 | 73 | elif status['type']=='current': |
| 74 | # Operation running | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
75 | progress='' |
|
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
76 | if 'progress_current' in status and 'progress_total' in status: |
|
19
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
77 | progress=' %d%%' % (status.progress_current/status.progress_total) |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
78 | elif 'original_size' in status and 'deduplicated_size' in status: |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
79 | progress=' %s O %s D' % (humanbytes(status.original_size), |
|
f9ce2442f14f
Display original/deduplicatd size if no progress percentage available
Tuomo Valkonen <tuomov@iki.fi>
parents:
18
diff
changeset
|
80 | humanbytes(status.deduplicated_size)) |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
81 | title="%s (running: %s%s)" % (status['name'], status['operation'], progress) |
| 18 | 82 | state=ACTIVE |
| 10 | 83 | else: # status['type']=='nothing': |
| 84 | # Should be unscheduled, nothing running | |
|
11
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
85 | title=status['name'] |
|
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
86 | |
| 15 | 87 | if status['errors']: |
| 88 | state=ERRORS | |
| 89 | ||
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
90 | return title, state |
| 10 | 91 | |
|
1
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
92 | |
|
4cdc9c1f6b28
basic scheduler structure draft, etc.
Tuomo Valkonen <tuomov@iki.fi>
parents:
diff
changeset
|
93 | class BorgendTray(rumps.App): |
|
9
aa121291eb0e
Rumps/Mac UI stuff is fucked and disables ^C etc.; threading doesn't help
Tuomo Valkonen <tuomov@iki.fi>
parents:
1
diff
changeset
|
94 | def __init__(self, name, backups): |
| 10 | 95 | self.lock=Lock() |
| 96 | self.backups=backups | |
| 97 | self.statuses=[None]*len(backups) | |
| 98 | ||
| 99 | with self.lock: | |
| 100 | # Initialise reporting callbacks and local status copy | |
| 101 | # (since rumps doesn't seem to be able to update menu items | |
| 102 | # without rebuilding the whole menu, and we don't want to lock | |
| 103 | # when calling Backup.status(), especially as we will be called | |
| 104 | # from Backup reporting its status, we keep a copy to allow | |
| 105 | # rebuilding the entire menu | |
| 106 | for index in range(len(backups)): | |
| 107 | b=backups[index] | |
| 108 | # Python closures suck dog's balls; hence the _index=index hack | |
| 109 | # See also http://math.andrej.com/2009/04/09/pythons-lambda-is-broken/ | |
| 110 | cb=(lambda obj, status, _index=index: | |
| 111 | self.__status_callback(obj, _index, status)) | |
| 112 | b.set_status_update_callback(cb) | |
| 113 | self.statuses[index]=b.status() | |
| 114 | ||
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
115 | menu, state=self.__rebuild_menu() |
|
9
aa121291eb0e
Rumps/Mac UI stuff is fucked and disables ^C etc.; threading doesn't help
Tuomo Valkonen <tuomov@iki.fi>
parents:
1
diff
changeset
|
116 | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
117 | super().__init__(traynames[state], menu=menu, quit_button=None) |
| 10 | 118 | |
| 119 | def __rebuild_menu(self): | |
| 120 | menu=[] | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
121 | state=INACTIVE |
| 10 | 122 | for index in range(len(self.backups)): |
| 123 | b=self.backups[index] | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
124 | title, this_state=make_title(self.statuses[index]) |
| 10 | 125 | logging.info('TITLE: %s' % title) |
| 126 | # Python closures suck dog's balls... | |
| 127 | # first and the last program I write in Python until somebody | |
| 128 | # fixes this brain damage | |
| 129 | cbm=lambda sender, _b=b: self.__menu_select_backup(sender, _b) | |
| 130 | item=rumps.MenuItem(title, callback=cbm) | |
| 131 | menu.append(item) | |
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
132 | state=combine_state(state, this_state) |
| 10 | 133 | |
| 134 | menu_quit=rumps.MenuItem("Quit...", callback=self.my_quit) | |
| 135 | menu.append(menu_quit) | |
| 136 | ||
|
14
5a988a2c2624
UI: progress percentange support (Borg doesn't seem to be reporting) + error indicator in tray: B?
Tuomo Valkonen <tuomov@iki.fi>
parents:
11
diff
changeset
|
137 | return menu, state |
| 10 | 138 | |
| 139 | ||
| 140 | def my_quit(self, _): | |
|
9
aa121291eb0e
Rumps/Mac UI stuff is fucked and disables ^C etc.; threading doesn't help
Tuomo Valkonen <tuomov@iki.fi>
parents:
1
diff
changeset
|
141 | rumps.quit_application() |
|
aa121291eb0e
Rumps/Mac UI stuff is fucked and disables ^C etc.; threading doesn't help
Tuomo Valkonen <tuomov@iki.fi>
parents:
1
diff
changeset
|
142 | |
| 10 | 143 | def __menu_select_backup(self, sender, b): |
| 144 | #sender.state=not sender.state | |
| 145 | logging.debug("Manually backup '%s'", b.name) | |
| 146 | b.create(None) | |
|
9
aa121291eb0e
Rumps/Mac UI stuff is fucked and disables ^C etc.; threading doesn't help
Tuomo Valkonen <tuomov@iki.fi>
parents:
1
diff
changeset
|
147 | |
| 10 | 148 | def __status_callback(self, obj, index, status): |
| 149 | logging.debug('Status callbackup %s' % str(status)) | |
| 150 | with self.lock: | |
| 151 | self.statuses[index]=status | |
| 152 | logging.debug('Rebuilding menu') | |
| 153 | self.menu.clear() | |
|
11
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
154 | menu, active=self.__rebuild_menu() |
|
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
155 | self.menu.update(menu) |
|
0bff53095f28
New tray title: B. or B! depending on activity
Tuomo Valkonen <tuomov@iki.fi>
parents:
10
diff
changeset
|
156 | self.title=traynames[active] |
| 10 | 157 | |
| 158 | ||
| 159 |