feat: add password generation and basic transient menu
This commit is contained in:
parent
28c01be6e3
commit
fb6fbad1fc
2
Eask
2
Eask
@ -3,7 +3,7 @@
|
||||
"Password generator for Emacs")
|
||||
|
||||
(website-url "https://labs.phundrak.com/phundrak/password-gen.el")
|
||||
(keywords "")
|
||||
(keywords "convenience")
|
||||
|
||||
(package-file "password-gen.el")
|
||||
|
||||
|
144
password-gen.el
144
password-gen.el
@ -6,8 +6,8 @@
|
||||
;; Maintainer: Lucien Cartier-Tilet <lucien@phundrak.com>
|
||||
;; URL: https://labs.phundrak.com/phundrak/password-gen.el
|
||||
;; Version: 0.1.0
|
||||
;; Package-Requires: ((emacs "26.1"))
|
||||
;; Keywords:
|
||||
;; Package-Requires: ((emacs "26.1") (transient "0.3.7"))
|
||||
;; Keywords: convenience
|
||||
|
||||
;; This file is not part of GNU Emacs.
|
||||
|
||||
@ -26,12 +26,148 @@
|
||||
|
||||
;;; Commentary:
|
||||
;;
|
||||
;; Password generator for Emacs
|
||||
;; Password generator for Emacs, based on transient.
|
||||
;;
|
||||
|
||||
;;; Code:
|
||||
|
||||
;; Happy coding! ;)
|
||||
;;; Requires
|
||||
(require 'transient)
|
||||
|
||||
(defconst password-gen-version "0.1.0")
|
||||
|
||||
(defgroup password-gen ()
|
||||
"Password generator for Emacs"
|
||||
:group 'applications
|
||||
:prefix "password-gen-"
|
||||
:link '(url-link :tag "Gitea" "https://labs.phundrak.com/phundrak/password-gen.el"))
|
||||
|
||||
|
||||
;;; Internal Functions
|
||||
|
||||
(eval-when-compile
|
||||
(defmacro password-gen--define-infix (key name description type version
|
||||
default &rest reader)
|
||||
"Define infix and its corresponding variable at once.
|
||||
The variable is named password-gen--NAME and is of type TYPE, has
|
||||
a DESCRIPTION and a specified VERSION. They KEY and READER are
|
||||
for the infix declaration.
|
||||
|
||||
This macro is largely copied from Tecosaur’s screenshot.el"
|
||||
(let ((var-name (concat "password-gen--" name)))
|
||||
`(progn
|
||||
(defcustom ,(intern var-name) ,default
|
||||
,description
|
||||
:type ,type
|
||||
:group 'password-gen
|
||||
:version ,version)
|
||||
(transient-define-infix ,(intern (concat "password-gen--set-" name)) ()
|
||||
"Set password-gen options."
|
||||
:class 'transient-lisp-variable
|
||||
:variable ',(intern var-name)
|
||||
:key ,key
|
||||
:description ,description
|
||||
:argument ,(concat "--" name)
|
||||
:reader (lambda (&rest _) ,@reader)))))
|
||||
|
||||
(password-gen--define-infix
|
||||
"-n" "numbers" "Include 0-9 numbers"
|
||||
'boolean "0.1.0" t
|
||||
(not password-gen--numbers))
|
||||
|
||||
(password-gen--define-infix
|
||||
"-C" "uppercase" "Include A-Z characters"
|
||||
'boolean "0.1.0" t
|
||||
(not password-gen--uppercase))
|
||||
|
||||
(password-gen--define-infix
|
||||
"-c" "lowercase" "Include a-z characters"
|
||||
'boolean "0.1.0" t
|
||||
(not password-gen--lowercase))
|
||||
|
||||
(password-gen--define-infix
|
||||
"-s" "symbols" "Include !@#$%^&* symbols"
|
||||
'boolean "0.1.0" t
|
||||
(not password-gen--symbols))
|
||||
|
||||
(password-gen--define-infix
|
||||
"-l" "length" "Length of the password"
|
||||
'boolean "0.1.0" 16
|
||||
(read-number "Length of the password: ")))
|
||||
|
||||
(defun password-gen--generate-password ()
|
||||
"Generate a password based on user preferences."
|
||||
(shell-command-to-string
|
||||
(format "tr -dc '%s%s%s%s' < /dev/urandom | head -c %d"
|
||||
(if password-gen--uppercase "A-Z" "")
|
||||
(if password-gen--lowercase "a-z" "")
|
||||
(if password-gen--numbers "0-9" "")
|
||||
(if password-gen--symbols "!@#$%^&*")
|
||||
password-gen--length)))
|
||||
|
||||
(defun password-gen--copy-generated-password ()
|
||||
"Generate new password and add it to kill ring."
|
||||
(kill-new (password-gen--generate-password)))
|
||||
|
||||
(defun password-gen--insert-generated-password ()
|
||||
"Insert at cursor a newly generated password."
|
||||
(insert (password-gen--generate-password)))
|
||||
|
||||
|
||||
;;; Transient Actions
|
||||
(eval-when-compile
|
||||
(defmacro password-gen--def-action (name description transient &rest body)
|
||||
"Create a function called from TRANSIENT.
|
||||
DESCRIPTION is the docstring of the function named
|
||||
password-gen--NAME, and it gets BODY as its body."
|
||||
`(defun ,(intern (concat "password-gen--action-" name)) (&optional _args)
|
||||
,(concat description "
|
||||
|
||||
This function is meant to be called by a transient.")
|
||||
(interactive
|
||||
(list (transient-args ,transient)))
|
||||
,@body))
|
||||
|
||||
(password-gen--def-action
|
||||
"quit"
|
||||
"Exit password generator"
|
||||
'password-gen-main
|
||||
#'nil)
|
||||
|
||||
(password-gen--def-action
|
||||
"copy"
|
||||
"Generate and copy password in kill-ring."
|
||||
'password-gen-main
|
||||
(password-gen--copy-generated-password))
|
||||
|
||||
(password-gen--def-action
|
||||
"insert"
|
||||
"Generate and insert password at cursor."
|
||||
'password-gen-main
|
||||
(password-gen--insert-generated-password)))
|
||||
|
||||
|
||||
;;; Transient Prefixes
|
||||
(transient-define-prefix password-gen-main ()
|
||||
["Options"
|
||||
(password-gen--set-length)
|
||||
(password-gen--set-uppercase)
|
||||
(password-gen--set-lowercase)
|
||||
(password-gen--set-numbers)
|
||||
(password-gen--set-symbols)]
|
||||
["Actions"
|
||||
("c" "Copy" password-gen--action-copy)
|
||||
("i" "Insert" password-gen--action-insert)
|
||||
("q" "Quit" password-gen--action-quit)]
|
||||
(interactive)
|
||||
(transient-setup 'password-gen))
|
||||
|
||||
|
||||
;;;###autoload
|
||||
(defun password-gen ()
|
||||
"Call the main transient for password-gen."
|
||||
(interactive)
|
||||
(call-interactively #'password-gen-main))
|
||||
|
||||
(provide 'password-gen)
|
||||
;;; password-gen.el ends here
|
||||
|
Loading…
Reference in New Issue
Block a user