summaryrefslogtreecommitdiff
path: root/.config/emacs/modules/bd--gpg.el
blob: f3d55cbca5dd1281113256029a9698a07a9c0d14 (plain)
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
;;; -*- lexical-binding: t; -*-
;;; Commentary:
;;; Code:


(require 'f)

(defvar bd/password-store-kill-ring-pointer nil
  "The tail of the kill ring whose car is the password.")
(defvar bd/password-store-time-before-clear 10
  "The time before a killed password is cleared.")

;;;; passwords
(defun bd/password-store-list ()
  "List password-store entries."
  (mapcar (lambda (file)
            (f-no-ext (f-relative file "~/.password-store/")))
          (f-files "~/.password-store" (lambda (file) (equal (f-ext file) "gpg")) t)))

(defun bd/password-store-clear (id)
  "Clears the most recent password copied to the kill ring."
  (when bd/password-store-kill-ring-pointer
    (setcar bd/password-store-kill-ring-pointer nil)
    (kill-new "")
    (setq bd/password-store-kill-ring-pointer nil)
    (message "Cleared password for %s from the kill ring and system clipboard." id)))

(defun bd/read-password (id)
  "Read the password-store entry corresponding to ID."
  (bd/password-store-clear "id")
  (let ((find-file-hook (remq 'recentf-track-opened-file find-file-hook)))
    (find-file (concat "~/.password-store/" id ".gpg"))
    (goto-char 1)
    (kill-new (buffer-substring-no-properties (line-beginning-position) (line-end-position)))
    (setq bd/password-store-kill-ring-pointer kill-ring-yank-pointer)
    (kill-buffer (current-buffer))
    (run-at-time bd/password-store-time-before-clear nil
                 (lambda () (funcall #'bd/password-store-clear id)))
    (message "Copied password for %s to the kill ring and system clipboard. Will clear in %s seconds."
             id bd/password-store-time-before-clear)))

(defun bd/selector-passwords ()
  "Selector source for password-store passwords."
  (selector-source-create
   "Passwords"
   :candidates
   (bd/password-store-list)
   :actions
   (list #'bd/read-password)))

(defun bd/password ()
  "Interactively select a password-store password."
  (interactive)
  (selector (list (bd/selector-passwords))))

;; pinentry
(use-package pinentry
  :config
  (pinentry-start)
  :custom
  (enable-recursive-minibuffers t)
  (pinentry-popup-prompt-window nil))

(defun pinentry-toggle ()
  "Stops and starts Pinentry service. Workaround
for a bug I've encountered."
  (interactive)
  (pinentry-stop)
  (pinentry-start))


(provide 'bd--gpg)
;;; bd-gpg ends here