2020-04-11 12:49:04 +02:00
|
|
|
# pylint: disable=C0111,R0903
|
|
|
|
|
2020-04-11 12:49:25 +02:00
|
|
|
"""Copy passwords from a password store into the clipboard (currently supports only 'pass')
|
2020-04-11 12:49:04 +02:00
|
|
|
|
|
|
|
Many thanks to [@bbernhard](https://github.com/bbernhard) for the idea!
|
|
|
|
|
2020-07-18 08:23:28 +02:00
|
|
|
Requires the following executable:
|
|
|
|
* pass (aka password-store)
|
|
|
|
|
2020-04-11 12:49:04 +02:00
|
|
|
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)
|
2020-07-25 17:46:57 +02:00
|
|
|
* vault.leave_menu: Boolean flag to close menu when the mouse leaves (defaults to False)
|
|
|
|
* vault.text: Text to display on the widget (defaults to <click-for-password>)
|
2020-05-08 20:58:35 +02:00
|
|
|
|
|
|
|
Many thanks to `bbernhard <https://github.com/bbernhard>`_ for the idea!
|
2020-04-11 12:49:04 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
|
2020-05-03 11:15:52 +02:00
|
|
|
# TODO:
|
2020-04-11 12:49:04 +02: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 13:35:12 +02:00
|
|
|
|
|
|
|
import core.module
|
|
|
|
import core.widget
|
|
|
|
import core.input
|
|
|
|
import core.event
|
|
|
|
|
|
|
|
import util.cli
|
|
|
|
import util.popup
|
2020-04-11 12:49:04 +02:00
|
|
|
|
2020-05-03 11:15:52 +02:00
|
|
|
|
2020-05-28 20:58:56 +02:00
|
|
|
def generate_callback(callback, path, name):
|
|
|
|
return lambda: callback(os.path.join(path, name))
|
|
|
|
|
2020-04-11 12:49:04 +02:00
|
|
|
def build_menu(parent, current_directory, callback):
|
|
|
|
with os.scandir(current_directory) as it:
|
|
|
|
for entry in it:
|
2020-05-03 11:15:52 +02:00
|
|
|
if entry.name.startswith("."):
|
|
|
|
continue
|
2020-04-11 12:49:04 +02:00
|
|
|
if entry.is_file():
|
2020-05-03 11:15:52 +02:00
|
|
|
name = entry.name[: entry.name.rfind(".")]
|
|
|
|
parent.add_menuitem(
|
|
|
|
name,
|
2020-05-28 20:58:56 +02:00
|
|
|
callback=generate_callback(callback, current_directory, name),
|
2020-05-03 11:15:52 +02:00
|
|
|
)
|
2020-04-11 12:49:04 +02:00
|
|
|
|
|
|
|
else:
|
2020-05-30 10:02:12 +02:00
|
|
|
submenu = util.popup.menu(parent, leave=False)
|
2020-05-03 11:15:52 +02:00
|
|
|
build_menu(
|
|
|
|
submenu, os.path.join(current_directory, entry.name), callback
|
|
|
|
)
|
2020-04-11 12:49:04 +02:00
|
|
|
parent.add_cascade(entry.name, submenu)
|
|
|
|
|
2020-05-03 11:15:52 +02:00
|
|
|
|
2020-04-11 13:35:12 +02:00
|
|
|
class Module(core.module.Module):
|
2020-04-26 16:39:24 +02:00
|
|
|
def __init__(self, config, theme):
|
|
|
|
super().__init__(config, theme, core.widget.Widget(self.text))
|
2020-04-11 13:35:12 +02:00
|
|
|
|
2020-05-03 11:15:52 +02: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 13:35:12 +02:00
|
|
|
self.__reset()
|
2020-05-03 11:15:52 +02:00
|
|
|
core.input.register(self, button=core.input.LEFT_MOUSE, cmd=self.popup)
|
2020-04-11 12:49:04 +02:00
|
|
|
|
|
|
|
def popup(self, widget):
|
2020-07-25 17:46:57 +02:00
|
|
|
menu = util.popup.menu(leave=util.format.asbool(self.parameter("leave_menu", False)))
|
2020-04-11 13:35:12 +02:00
|
|
|
|
|
|
|
build_menu(menu, self.__path, self.__callback)
|
|
|
|
menu.show(widget, offset_x=self.__offx, offset_y=self.__offy)
|
|
|
|
|
|
|
|
def __reset(self):
|
|
|
|
self.__timer = None
|
2020-05-03 11:15:52 +02:00
|
|
|
self.__text = str(self.parameter("text", "<click-for-password>"))
|
2020-04-11 13:35:12 +02:00
|
|
|
|
|
|
|
def __callback(self, secret_name):
|
2020-05-03 11:15:52 +02:00
|
|
|
secret_name = secret_name.replace(self.__path, "") # remove common path
|
2020-04-11 13:35:12 +02:00
|
|
|
if self.__timer:
|
|
|
|
self.__timer.cancel()
|
2020-05-28 20:58:56 +02:00
|
|
|
env = os.environ
|
|
|
|
env["PASSWORD_STORE_CLIP_TIME"] = str(self.__duration)
|
2020-05-03 11:15:52 +02:00
|
|
|
res = util.cli.execute(
|
2020-05-28 20:58:56 +02:00
|
|
|
"pass show -c {}".format(secret_name),
|
2020-05-03 11:15:52 +02:00
|
|
|
wait=False,
|
2020-05-28 20:58:56 +02:00
|
|
|
env=env,
|
2020-05-30 10:02:12 +02:00
|
|
|
ignore_errors=True,
|
2020-05-03 11:15:52 +02:00
|
|
|
)
|
2020-04-11 13:35:12 +02:00
|
|
|
self.__timer = threading.Timer(self.__duration, self.__reset)
|
|
|
|
self.__timer.start()
|
|
|
|
self.__start = int(time.time())
|
|
|
|
self.__text = secret_name
|
2020-04-11 12:49:04 +02:00
|
|
|
|
|
|
|
def text(self, widget):
|
2020-04-11 13:35:12 +02:00
|
|
|
if self.__timer:
|
2020-05-03 11:15:52 +02:00
|
|
|
return "{} ({}s)".format(
|
|
|
|
self.__text, self.__duration - (int(time.time()) - self.__start)
|
|
|
|
)
|
2020-04-11 13:35:12 +02:00
|
|
|
return self.__text
|
2020-04-11 12:49:04 +02:00
|
|
|
|
2020-05-03 11:15:52 +02:00
|
|
|
|
2020-04-11 12:49:04 +02:00
|
|
|
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
|