dispass.el/dispass.el
Tom Willemsen d547d846c6 Add the -l switch functionality
Using a numeric prefix argument when calling either `dispass' or
`dispass-create' will invoke dispass with the -l switch to cut
passwords down to a certain size.
2012-06-17 14:44:46 +02:00

155 lines
5.2 KiB
EmacsLisp

;;; dispass.el --- Generate and disperse/dispell passwords
;; Copyright (C) 2012 Tom Willemsen <tom@ryuslash.org>
;; Author: Tom Willemsen <tom@ryuslash.org>
;; Created: Jun 8, 2012
;; Version: 0.1a7.3
;; Keywords: encryption, security
;; Permission to use, copy, modify, and distribute this software for any
;; purpose with or without fee is hereby granted, provided that the
;; above copyright notice and this permission notice appear in all
;; copies.
;; THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
;; WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
;; WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
;; AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR
;; CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
;; OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
;; NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
;; CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
;;; Commentary:
;; dispass.el is an emacs wrapper around dispass
;; (http://dispass.babab.nl).
;;; Installation:
;; Place this file somewhere on your filesystem, either in your
;; `load-path' or somewhere else which you will have to add to your
;; `load-path', like so:
;; (add-to-list 'load-path "/location/of/dispass.el")
;; And then `load', `require' or `autoload' it in your emacs init
;; file, for example:
;; (require 'dispass)
;; _Or_ if you have package.el you could use `package-install-file'.
;;; Usage:
;; Using dispass.el is simple, once installed. Either call `dispass'
;; to recall a priviously generated password or call `dispass-create'
;; to generate a new password.
;; The only real difference between the two is that `dispass-create'
;; asks to confirm the password. Both will ask for a label.
;; When a numeric argument is used when calling either
;; `dispass-create' or `dispass', that argument is sent to the dispass
;; program along with the -l switch. This cuts the length of the
;; password to that many characters. For example:
;; C-5 M-x dispass<RET>
;; will generate a password of 5 characters for label "test" and
;; password "test".
;; Once a password has been generated it is inserted into the kill
;; ring and the system's clipboard so it can be easily inserted into
;; password field, this makes the generated password easy to see in
;; plaintext in the `kill-ring' variable, though.
;;; Change Log:
;; 0.1a7 - Initial release.
;; 0.1a7.1 - Don't show password, copy directly into the clipboard.
;; 0.1a7.2 - Kill buffer whenever we're finished with it.
;; 0.1a7.3 - Add the possility to cut passwords short by using a
;; numeric prefix argument.
;; - Add `dispass-executable' which holds the location of the
;; dispass executable script.
;;; Code:
(defvar dispass-executable "dispass"
"The location of the dispass executable")
(defun dispass-process-sentinel (proc status)
"Report PROC's status change to STATUS."
(let ((status (substring status 0 -1))
(buffer (process-buffer proc)))
(unless (string-equal status "finished")
(message "dispass %s" status))
(unless (eq (current-buffer) proc)
(kill-buffer buffer))))
(defun dispass-erase-buffer (buffer)
"Completely erase the contents of BUFFER"
(save-current-buffer
(set-buffer buffer)
(buffer-disable-undo buffer)
(kill-buffer buffer)))
(defun dispass-process-filter-for (label)
"Create a function that will process any lines whilst keeping
an eye out for LABEL."
`(lambda (proc string)
"Process STRING coming from PROC."
(cond ((string-match "^\\(Password\\|Again\\): ?$" string)
(process-send-string proc
(concat (read-passwd string nil) "\n")))
((string-match (concat "^[ \t]*" ,label "[ \t]*\\(.+\\)$")
string)
(let ((buffer (process-buffer proc)))
(with-current-buffer buffer
(insert (match-string 1 string))
(clipboard-kill-ring-save (point-min) (point-max))
(message "Password copied to clipboard.")
(unless (eq (process-status proc) 'run)
(kill-buffer buffer))))))))
(defun dispass-start-process (label create length)
"Start dispass process. When CREATE is non-nil send along the
-c switch to make it ask for a password twice. When LENGTH is
an integer and greater than 0, send along the -l switch with
LENGTH."
(let ((args `("-o" ,label))
proc)
(when create
(setq args (append '("-c") args)))
(when (and (integerp length) (> length 0))
(setq args (append `("-l" ,(number-to-string length)) args)))
(setq proc (apply 'start-process "dispass" "*dispass*"
dispass-executable args))
(set-process-sentinel proc 'dispass-process-sentinel)
(set-process-filter proc (dispass-process-filter-for label))))
;;;###autoload
(defun dispass-create (label &optional length)
(interactive "MLabel: \nP")
"Create a new password for LABEL."
(dispass-start-process label t length))
;;;###autoload
(defun dispass (label &optional length)
(interactive "MLabel: \nP")
"Recreate a password previously used."
(dispass-start-process label nil length))
(provide 'dispass)
;;; dispass.el ends here