forked from xlucn/cpass
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcpass.py
executable file
·727 lines (617 loc) · 27.3 KB
/
cpass.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
#!/usr/bin/env python3
# Author: Lu Xu <oliver_lew at outlook dot com>
# License: MIT License Copyright (c) 2021 Lu Xu
import os
import re
import urwid
import logging
import tempfile
import configparser
from subprocess import run, PIPE, DEVNULL
version = "0.9.4"
class PassNode(urwid.AttrMap):
def __init__(self, node, root, isdir=False):
self.empty = node is None
self.isdir = isdir
self.node = node or "-- EMPTY --"
self.path = os.path.join(root, node) if node else ''
self.icon = config.icon_dir if isdir else config.icon_file if node else ''
self._selectable = True
super().__init__(urwid.Columns([
('pack', urwid.Text(self.icon)),
urwid.Text(self.node, wrap='clip'),
('pack', urwid.Text(''))
]),
'dir' if isdir else '' if node else 'bright',
'focusdir' if isdir else 'focus' if node else 'bright',
)
self.update_count()
def update_count(self):
# 'topdown' option in os.walk makes this possible (see Pass.extract_all),
# so that children folders are traversed before its parent (the 'len'
# function below can be executed).
if self.isdir:
count = len(Pass.all_pass[self.path])
self.original_widget.contents[2][0].set_text(str(count))
def keypress(self, size, key):
""" let the widget pass through the keys to parent widget """
return key
class PassList(urwid.ListBox):
def __init__(self, body, root='', ui=None):
self._ui = ui
self.root = root
self._size = (1, 1)
super().__init__(body)
def mouse_event(self, size, event, button, col, row, focus):
self._size = size
focus_offset = self.get_focus_offset_inset(size)[0]
logging.debug("passlist mouse event: {} {} {} {} {} {} {} {}".format(
size, event, button, col, row, focus, self.focus_position, focus_offset
))
if button == 1:
if size[1] > len(self.body):
# NOTE: offset is wrong(?) when size is larger than length
# so the processing is different
if row == self.focus_position:
self.dir_navigate('down')
else:
self.list_navigate(new_focus=row)
else:
if row == focus_offset:
self.dir_navigate('down')
else:
self.list_navigate(new_focus=self.focus_position - focus_offset + row)
elif button == 3:
self.dir_navigate('up')
elif button == 4:
self.list_navigate(-1)
elif button == 5:
self.list_navigate(1)
else:
return super().mouse_event(size, event, button, col, row, focus)
def keypress(self, size, key):
self._size = size
logging.debug("passlist keypress: {} {}".format(key, size))
list_navigation_offsets = {
'down': 1,
'up': -1,
# overshoot to go to bottom/top
'end': len(self.body),
'home': -len(self.body),
'down_screen': size[1],
'up_screen': -size[1],
'down_half_screen': size[1] // 2,
'up_half_screen': -size[1] // 2,
}
dir_navigation_directions = {
# the confirm key doubles as enter folder key
'confirm': 'down',
'dir_down': 'down',
'dir_up': 'up',
}
action = config.keybindings.get(key)
if action in list_navigation_offsets:
self.list_navigate(list_navigation_offsets[action])
elif action in dir_navigation_directions:
self.dir_navigate(dir_navigation_directions[action])
else:
return super().keypress(size, key)
def dir_navigate(self, direction):
# record current position
Pass.all_pass[self.root].pos = self.focus_position
# change root position accordingly
if direction in 'down' and self.focus.isdir:
self.root = os.path.join(self.root, self.focus.node)
elif direction in 'up':
self.root = os.path.dirname(self.root)
# update listbox content, this way the list itself is not replaced
self.body[:] = Pass.all_pass[self.root]
# restore cursor position of the new root
self.focus_position = Pass.all_pass[self.root].pos
self._ui.update_view()
def list_navigate(self, shift=0, new_focus=None):
""" either specify a shift offset, or an absolute target position """
offset = self.get_focus_offset_inset(self._size)[0]
if new_focus is not None:
shift = new_focus - self.focus_position
else:
new_focus = shift + self.focus_position
new_offset = offset + shift
# border check
new_focus = min(max(0, new_focus), len(self.body) - 1)
new_offset = min(max(0, new_offset), self._size[1] - 1)
self.change_focus(self._size, new_focus, offset_inset=new_offset)
self._ui.update_view()
def insert(self, node):
def insert_relative(r, n):
# if starts with /, then assume the node is relative to store root
if n.startswith('/'):
n = n.lstrip('/')
r = ''
# separate at the first /
n1, sep, n2 = n.partition('/')
# recursively insert if there are more levels
if sep == '/':
insert_relative(os.path.join(r, n1), n2)
# change stored list
passnode = PassNode(n1, r, sep == '/')
if Pass.all_pass.get(r) is None:
Pass.all_pass[r] = FolderWalker(r)
Pass.all_pass[r].pos = Pass.all_pass[r].insert(passnode)
# do not change cursor position if the path is not relative
return Pass.all_pass[r].pos if r == self.root else None
inserted_pos = insert_relative(self.root, node.strip())
# change listwalker
self.body[:] = Pass.all_pass[self.root]
# focus the new node
self.list_navigate(new_focus=inserted_pos)
self._ui.update_view()
def delete(self, pos):
# change stored list
Pass.all_pass[self.root].pop(pos)
# change listwalker
self.body[:] = Pass.all_pass[self.root]
self._ui.update_view()
# TODO: this seems odd being here
def update_root_count(self):
for n in Pass.all_pass[os.path.dirname(self.root)]:
if n.node == self.root and n.isdir:
n.update_count()
return
class FolderWalker(list):
"""
Customize list operations, mainly
- keep a placeholder item in empty list and
- keep items sorted
"""
def __init__(self, root, dirs=[], files=[]):
self.pos = 0 # cursor position
self[:] = [PassNode(f, root, True) for f in sorted(dirs, key=str.lower)] + \
[PassNode(f, root) for f in sorted(files, key=str.lower)]
# prevent empty list, which troubles listbox operations
if len(self) == 0:
super().append(PassNode(None, None))
def pop(self, index=-1):
super().pop(index)
# add a empty placeholder
if len(self) == 0:
super().append(PassNode(None, None))
def insert(self, node):
# if node already exist, return the index
for n in self:
if n.node == node.node and n.isdir == node.isdir:
return self.index(n)
# pop the empty placeholder node beforehand
if len(self) == 1 and self[0].empty:
super().pop()
# insert and sort, with directories sorted before files
super().insert(self.pos, node)
self[:] = sorted([n for n in self if n.isdir], key=lambda n: n.node.lower()) + \
sorted([n for n in self if not n.isdir], key=lambda n: n.node.lower())
return self.index(node)
# TODO: auto change split direction based on terminal size
# TODO: multiline insert, this should be easy since we have the workaround in Pass.edit
# TODO: mv, cp support
# TODO: QR code generate, maybe?
# TODO: background preview, or/and cache preview results
# TODO: CLI arguments
# TODO: git support
# TODO: otp support
class UI(urwid.Frame):
def __init__(self):
self._app_string = 'cPass'
self._help_string = ' a:generate d:delete e:edit i:insert y:copy z:toggle /:search'
self._edit_type = None
self._last_preview = None
self._preview_shown = True
self._search_pattern = None
self._search_direction = 1
# header
self.header_prefix = urwid.Text(('border', '{}:'.format(self._app_string)))
self.path_indicator = urwid.Text(('bright', ''), wrap='clip')
self.help_text = urwid.Text(self._help_string, wrap='clip', align='right')
# priority on showing full path
self.header_widget = urwid.Columns([
('pack', self.header_prefix),
('pack', self.path_indicator),
self.help_text
], dividechars=1)
# footer
self.messagebox = urwid.Text('')
self.count_indicator = urwid.Text('', align='right')
self.footer_widget = urwid.Columns([
self.messagebox,
('pack', urwid.AttrMap(self.count_indicator, 'border'))
])
# some dynamic widgets
self.divider = urwid.AttrMap(urwid.Divider('-'), 'border')
self.preview = urwid.Filler(urwid.Text(''), valign='top')
self.editbox = urwid.Edit()
self.walker = urwid.SimpleListWalker(Pass.all_pass[''])
self.listbox = PassList(self.walker, ui=self)
# use Columns for horizonal layout, and Pile for vertical
if config.preview_layout in ['side', 'horizontal']:
self.middle = urwid.Columns([], dividechars=1)
elif config.preview_layout in ['bottom', 'vertical']:
self.middle = urwid.Pile([])
self.update_preview_layout()
self.update_view()
super().__init__(self.middle, self.header_widget, self.footer_widget)
def message(self, message, alert=False):
self.messagebox.set_text(('alert' if alert else 'normal',
message.replace('\n', ' ')))
def update_preview_layout(self):
if self._preview_shown:
if config.preview_layout in ['side', 'horizontal']:
self.middle.contents = [(self.listbox, ('weight', 1, False)),
(self.preview, ('weight', 1, False))]
if config.preview_layout in ['bottom', 'vertical']:
self.middle.contents = [(self.listbox, ('weight', 1)),
(self.divider, ('pack', 1)),
(self.preview, ('weight', 1))]
self.update_preview()
else:
self.middle.contents = [(self.listbox, ('weight', 1, False))]
self.middle.focus_position = 0
def mouse_event(self, size, event, button, col, row, focus):
logging.debug(f"ui mouse event: {size} {event} {button} {col} {row} {focus}")
# Prevent focus change due to clicking when editing
r = self.contents['footer'][0].rows(size[:1], True)
if self._edit_type is None or self._edit_type and row >= size[1] - r:
super().mouse_event(size, event, button, col, row, focus)
def keypress(self, size, key):
logging.debug("ui keypress: {} {}".format(key, size))
action = config.keybindings.get(key)
if action == 'cancel':
self.unfocus_edit()
elif self._edit_type == "copy":
self.unfocus_edit()
self.copy_by_key(key)
elif self._edit_type == "delete":
self.unfocus_edit()
self.delete_confirm(key)
elif self._edit_type is not None:
if action == 'confirm':
self.handle_input()
else:
# pass through to edit widget (the focused widget)
return super().keypress(size, key)
elif action == 'quit':
raise urwid.ExitMainLoop
elif action == 'search' or action == 'search_back':
self.focus_edit("search", '/' if action == 'search' else '?')
self._search_direction = 1 if action == 'search' else -1
elif action == 'search_next' or action == 'search_prev':
self.search_in_dir(self._search_pattern,
1 if action == 'search_next' else -1)
elif action == 'insert':
self.focus_edit("insert", 'Enter password filename: ')
elif action == 'generate':
self.focus_edit("generate", 'Generate a password file: ')
elif action == 'edit' and not self.listbox.focus.isdir:
self.run_pass(Pass.edit, None,
self.listbox.focus.node, self.listbox.root, "Edit: {}")
urwid.emit_signal(self, 'redraw')
elif action == 'delete' and not self.listbox.focus.empty:
self.focus_edit("delete", 'Are you sure to delete {} {}? [Y/n]'.format(
"the whole folder" if self.listbox.focus.isdir else "the file",
os.path.join('/', self.listbox.root, self.listbox.focus.node)
))
elif action == 'copy':
self.copy_confirm()
elif action == 'toggle_preview':
self._preview_shown = not self._preview_shown
self.update_preview_layout()
else:
return super().keypress(size, key)
def unfocus_edit(self):
self._edit_type = None
self.contents['footer'] = (self.footer_widget, None)
self.set_focus('body')
self.messagebox.set_text('')
self.editbox.set_mask(None)
def focus_edit(self, edit_type, cap, mask=None):
self._edit_type = edit_type
self.contents['footer'] = (self.editbox, None)
self.set_focus('footer')
self.editbox.set_caption(cap)
self.editbox.set_mask(mask)
self.editbox.set_edit_text('')
def handle_input(self):
# these codes are ugly
edit_type = self._edit_type
self.unfocus_edit()
if edit_type == "search":
self._search_pattern = self.editbox.edit_text
self.search_in_dir(self._search_pattern, 1)
elif edit_type == "generate":
self.run_pass(Pass.generate, self.listbox.insert,
self.editbox.edit_text, self.listbox.root, "Generate: {}")
self.listbox.update_root_count()
elif edit_type == "insert":
self._insert_node = self.editbox.edit_text
self.focus_edit("insert_password", 'Enter password: ', mask='*')
elif edit_type == "insert_password":
self._insert_pass = self.editbox.edit_text
self.focus_edit("insert_password_confirm", 'Enter password again: ', mask='*')
elif edit_type == "insert_password_confirm":
self._insert_pass_again = self.editbox.edit_text
if self._insert_pass == self._insert_pass_again:
self.run_pass(Pass.insert, self.listbox.insert,
self._insert_node, self.listbox.root, "Insert: {}",
args=(self._insert_pass,))
self.listbox.update_root_count()
else:
self.message("Password is not the same", alert=True)
def update_view(self):
# update header
self.path_indicator.set_text(('bright', "/" + self.listbox.root))
# update footer
self.count_indicator.set_text("{}/{}".format(
self.listbox.focus_position + 1,
len(self.listbox.body)
) if not self.listbox.focus.empty else "0/0")
self.update_preview()
def update_preview(self, force=False):
if not self._preview_shown:
return
if not force and self.listbox.focus == self._last_preview:
return
self._last_preview = self.listbox.focus
if not self.listbox.focus.empty:
path = os.path.join(self.listbox.root, self.listbox.focus.node)
if self.listbox.focus.isdir:
preview = "\n".join([(f.icon + f.node) for f in Pass.all_pass[path]])
else:
res = Pass.show(path)
preview = res.stderr if res.returncode else res.stdout
else:
preview = ""
self.preview.original_widget.set_text(preview)
def run_pass(self, func, lfunc, node, root, msg='', args=(), largs=()):
# do not accept password name ends with /, pass itself has problems
if node.endswith('/'):
self.message(f'Can not create a directory: {node}.', alert=True)
return
path = os.path.join(root, node)
res = func(path, *args)
if res.returncode == 0:
self.message(msg.format(path) if func != Pass.show else '')
if lfunc:
lfunc(node if lfunc == self.listbox.insert else largs[0])
# some operations like generating password need updating the preview
self.update_preview(True)
else:
self.message(res.stderr, alert=True)
return res
def delete_confirm(self, key):
if key in ['y', 'Y', 'd', 'enter']:
self.run_pass(Pass.delete, self.listbox.delete,
self.listbox.focus.node, self.listbox.root,
"Deleting {}", largs=(self.listbox.focus_position,))
# TODO: put this into lower level functions
self.listbox.update_root_count()
elif key in ['n', 'N']:
self.message("Abort.")
else:
self.message("Invalid option.", alert=True)
def parse_pass(self, passwd):
# TODO: mark numbers on the side
"""
parse the decryped content of the password file
and relate shortcut keys to the corrsponding texts
"""
lines = passwd.split('\n')
# 1. default: yy to copy first line, ya to copy all lines
copiable_fields = {'a': passwd, 'y': lines[0], '1': lines[0]}
for i in range(1, len(lines)):
field, sep, value = [s.strip() for s in lines[i].partition(':')]
# 2. y[0-9] to copy that line, right of colon if applicable
if i < 10:
copiable_fields[str(i + 1)[-1]] = value if sep == ':' else field
# 3. customized field shortcuts
if sep == ':' and field in config.copy_bindings:
copiable_fields[config.copy_bindings[field]] = value
return copiable_fields
def copy_confirm(self):
if self.listbox.focus.isdir:
return
if self._preview_shown:
password = self.preview.original_widget.text
else:
res = self.run_pass(Pass.show, None, self.listbox.focus.node, self.listbox.root)
if res.returncode == 0:
password = res.stdout
else:
return
pw = self.parse_pass(password.rstrip('\n'))
self.focus_edit("copy", 'Copy [{}]: '.format(''.join(sorted(pw))))
self._parsed_password = pw
def copy_by_key(self, key):
if key in self._parsed_password:
copy_text = self._parsed_password[key]
# stderr and stdout have to be dropped, otherwise the program is stuck
res = run(['xclip', '-selection', Pass.X_SELECTION],
text=True, input=copy_text, stderr=DEVNULL, stdout=DEVNULL)
if res.returncode == 0:
self.message("Copied.")
else:
self.message("Copy with xclip failed", alert=True)
else:
self.message("Nothing copied", alert=True)
def search_in_dir(self, pattern, direction):
""" direction = 1 or -1 to specify the search direction """
if pattern is None:
self.message("No search pattern", alert=True)
return
# search from the next/previous, wrap if reaching bottom/top
start = self.listbox.focus_position
direction *= self._search_direction
# list of indexes according to the start point and order
indexes = list(range(len(self.listbox.body)))
# the math here is kind of magic, it's the result after simplification
search_list = (indexes[start+direction::direction] +
indexes[:start+direction:direction])
for i in search_list:
node = self.listbox.body[i].node
# ignore case if all letters are lower case
if pattern == pattern.lower():
node = node.lower()
# search for all space separated words
if all([s in node for s in pattern.split()]):
self.listbox.list_navigate(new_focus=i)
return
self.message("No matching", alert=True)
class Pass:
FALLBACK_PASS_DIR = os.path.join(os.getenv("HOME"), ".password_store")
PASS_DIR = os.getenv("PASSWORD_STORE_DIR", FALLBACK_PASS_DIR)
X_SELECTION = os.getenv("PASSWORD_STORE_X_SELECTION", "clipboard")
EDITOR = os.getenv("EDITOR", "vi")
all_pass = dict()
# exit if pass dir does not exit
if not os.path.exists(PASS_DIR):
print("'{}' or $PASSWORD_STORE_DIR does not exist".format(FALLBACK_PASS_DIR))
print("See `man pass` for how to set password storage directory.")
exit()
@classmethod
def extract_all(cls):
# pass files traversal, topdown option is essential, see PassNode
for root, dirs, files in os.walk(cls.PASS_DIR, topdown=False):
if not root.startswith(os.path.join(cls.PASS_DIR, '.git')):
root = os.path.normpath(os.path.relpath(root, cls.PASS_DIR)).lstrip('.')
dirs = [d for d in dirs if d != '.git']
files = [file[:-4] for file in files if file.endswith('.gpg')]
# NOTE: all_pass, FolderWalker, PassNode references are in a cycle.
cls.all_pass[root] = FolderWalker(root, dirs, files)
@staticmethod
def show(path):
logging.debug("Showing password for {}".format(path))
return run(['pass', 'show', path], stdout=PIPE, stderr=PIPE, text=True)
@classmethod
def edit(cls, path):
# work around terminal output by manually edit temp file and insert with multiline
with tempfile.NamedTemporaryFile() as fp:
res = cls.show(path)
if res.returncode != 0:
return res
fp.write(res.stdout.encode())
fp.flush()
# can not pipe stdout because editor won't show otherwise
res = run([cls.EDITOR, fp.name], stderr=PIPE)
if res.returncode != 0:
return res
fp.seek(0)
password = fp.read()
return run(['pass', 'insert', '-m', '-f', path], input=password,
stderr=PIPE, stdout=PIPE)
@staticmethod
def insert(path, password):
pw = password + '\n' + password + '\n'
return run(['pass', 'insert', '-f', path], input=pw,
stdout=PIPE, stderr=PIPE, text=True)
@staticmethod
def generate(path):
command = ['pass', 'generate', '-f', path]
if config.no_symbols:
command.append('-n')
return run(command, stdout=PIPE, stderr=PIPE, text=True)
@staticmethod
def delete(path):
command = ['pass', 'rm', '-r', '-f', path]
return run(command, stdout=PIPE, stderr=PIPE, text=True)
class MyConfigParser(configparser.RawConfigParser):
def __init__(self):
super().__init__()
DEFAULT_CONFIG_DIR = os.path.join(os.getenv("HOME"), ".config")
CONFIG_DIR = os.getenv("XDG_CONFIG_DIR", DEFAULT_CONFIG_DIR)
CONFIG = os.path.join(CONFIG_DIR, "cpass", "cpass.cfg")
if os.path.exists(CONFIG):
self.read(CONFIG)
self.preview_layout = self.get('ui', 'preview_layout', 'side')
self.icon_dir = self.get('icon', 'dir', '/')
self.icon_file = self.get('icon', 'file', ' ')
self.no_symbols = self.get('pass', 'no_symbols', 'false', boolean=True)
self.keybindings = self.get_keybindings()
self.palette = self.get_palette()
self.copy_bindings = self.get_copybindings()
def get(self, section, option, fallback=None, boolean=False):
try:
result = super().get(section, option)
return result == 'true' if boolean else result.strip("\"\'")
except (configparser.NoOptionError, configparser.NoSectionError):
return fallback
def get_keybindings(self):
action_keys = {
'dir_down': ['l', 'right'],
'dir_up': ['h', 'left'],
'down': ['j', 'down', 'ctrl n'],
'up': ['k', 'up', 'ctrl p'],
'down_screen': ['page down', 'ctrl f'],
'up_screen': ['page up', 'ctrl b'],
'down_half_screen': ['ctrl d'],
'up_half_screen': ['ctrl u'],
'end': ['G', 'end'],
'home': ['g', 'home'],
'cancel': ['esc'],
'confirm': ['enter'],
'search': ['/'],
'search_back': ['?'],
'search_next': ['n'],
'search_prev': ['N'],
'insert': ['i'],
'generate': ['a'],
'edit': ['e'],
'delete': ['d'],
'copy': ['y'],
'toggle_preview': ['z'],
'quit': ['q']
}
# map keys to actions, only one action can be mapped to one key
keys = {}
# default key bindings
for action in action_keys:
for key in action_keys[action]:
keys[key] = action
# update from configuration file
if self.has_section('keys'):
for action in self.options('keys'):
for key in re.split(',\\s*', self.get('keys', action, '')):
keys[key] = action
return keys
def get_palette(self):
palettes = {
# name fg bg mono
'normal': ('default', 'default'),
'border': ('light green', 'default'),
'dir': ('light blue', 'default'),
'alert': ('light red', 'default'),
'bright': ('white', 'default'),
'focus': ('standout', 'default'),
'focusdir': ('black', 'light blue', 'bold'),
}
# update from configuration file
if self.has_section('color'):
for name in self.options('color'):
palettes[name] = re.split(',\\s*', self.get('color', name, ''))
return [(name, *palettes[name]) for name in palettes]
def get_copybindings(self):
""" get field-key pairs """
copy_bindings = {'login': 'l'}
if self.has_section('copy_fields'):
for field in self.options('copy_fields'):
copy_bindings[field] = self.get('copy_fields', field)
return copy_bindings
def main():
Pass.extract_all()
passui = UI()
mainloop = urwid.MainLoop(passui, palette=config.palette)
# set no timeout after escape key
mainloop.screen.set_input_timeouts(complete_wait=0)
urwid.register_signal(UI, 'redraw')
urwid.connect_signal(passui, 'redraw', mainloop.screen.clear)
mainloop.run()
logging.basicConfig(level=(logging.DEBUG if os.getenv('DEBUG') else logging.DEBUG),
filename=os.path.join(tempfile.gettempdir(), 'cpass.log'))
config = MyConfigParser()
if __name__ == '__main__':
main()