bumblebee-status/modules/core/vault.py

99 lines
3.1 KiB
Python
Raw Normal View History

2020-04-11 10:49:04 +00:00
# pylint: disable=C0111,R0903
2020-04-11 10:49:25 +00:00
"""Copy passwords from a password store into the clipboard (currently supports only 'pass')
2020-04-11 10:49:04 +00:00
Many thanks to [@bbernhard](https://github.com/bbernhard) for the idea!
Parameters:
* vault.duration: Duration until password is cleared from clipboard (defaults to 30)
* vault.location: Location of the password store (defaults to ~/.password-store)
* vault.offx: x-axis offset of popup menu (defaults to 0)
* vault.offy: y-axis offset of popup menu (defaults to 0)
"""
# TODO:
2020-04-11 10:49:04 +00:00
# - support multiple backends by abstracting the menu structure into a tree
# - build the menu and the actions based on that abstracted tree
#
import os
import time
import threading
2020-04-11 11:35:12 +00:00
import core.module
import core.widget
import core.input
import core.event
import util.cli
import util.popup
2020-04-11 10:49:04 +00:00
2020-04-11 10:49:04 +00:00
def build_menu(parent, current_directory, callback):
with os.scandir(current_directory) as it:
for entry in it:
if entry.name.startswith("."):
continue
2020-04-11 10:49:04 +00:00
if entry.is_file():
name = entry.name[: entry.name.rfind(".")]
parent.add_menuitem(
name,
callback=lambda: callback(os.path.join(current_directory, name)),
)
2020-04-11 10:49:04 +00:00
else:
2020-04-11 11:35:12 +00:00
submenu = util.popup.menu(parent, leave=False)
build_menu(
submenu, os.path.join(current_directory, entry.name), callback
)
2020-04-11 10:49:04 +00:00
parent.add_cascade(entry.name, submenu)
2020-04-11 11:35:12 +00:00
class Module(core.module.Module):
def __init__(self, config, theme):
super().__init__(config, theme, core.widget.Widget(self.text))
2020-04-11 11:35:12 +00:00
self.__duration = int(self.parameter("duration", 30))
self.__offx = int(self.parameter("offx", 0))
self.__offy = int(self.parameter("offy", 0))
self.__path = os.path.expanduser(
self.parameter("location", "~/.password-store/")
)
2020-04-11 11:35:12 +00:00
self.__reset()
core.input.register(self, button=core.input.LEFT_MOUSE, cmd=self.popup)
2020-04-11 10:49:04 +00:00
def popup(self, widget):
2020-04-11 11:35:12 +00:00
menu = util.popup.menu(leave=False)
build_menu(menu, self.__path, self.__callback)
menu.show(widget, offset_x=self.__offx, offset_y=self.__offy)
def __reset(self):
self.__timer = None
self.__text = str(self.parameter("text", "<click-for-password>"))
2020-04-11 11:35:12 +00:00
def __callback(self, secret_name):
secret_name = secret_name.replace(self.__path, "") # remove common path
2020-04-11 11:35:12 +00:00
if self.__timer:
self.__timer.cancel()
res = util.cli.execute(
"pass -c {}".format(secret_name),
wait=False,
env={"PASSWORD_STORE_CLIP_TIME": self.__duration},
)
2020-04-11 11:35:12 +00:00
self.__timer = threading.Timer(self.__duration, self.__reset)
self.__timer.start()
self.__start = int(time.time())
self.__text = secret_name
2020-04-11 10:49:04 +00:00
def text(self, widget):
2020-04-11 11:35:12 +00:00
if self.__timer:
return "{} ({}s)".format(
self.__text, self.__duration - (int(time.time()) - self.__start)
)
2020-04-11 11:35:12 +00:00
return self.__text
2020-04-11 10:49:04 +00:00
2020-04-11 10:49:04 +00:00
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4