| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251 | ;;;(ql:quickload "cl-store")(ql:quickload "yason")(ql:quickload "cl-ppcre")(ql:quickload "unix-opts")(ql:quickload "ironclad");;; Features;;; - Import records from old .txt format;;; - Interactive prompt to manage expenses;;; - Generic expense handling;;; TODO;;; - Non interactive CLI Interface;;; - Upload/download support like perl version;;;   - Should support encryption/decryption of records;;;; Reimplementation of my bills tracker in Lisp;;; All records exist in this data structure;;; nil on start and loaded in from file;;; *records* represents as hash of months,;;; where the key is the month stamp, eg 20210701;;; and the value is the monthly expenses hash(defvar *records* (make-hash-table :test 'equalp))(defun file-test (filename)  (if (probe-file filename) filename (print "Couldn't find filename")));;; Used by "print-month" arg to validate;;; the user provided a valid key(defun check-month (month-key)  (if (stringp month-key) month-key))(opts:define-opts  (:name :help   :description "Print help text"   :short #\h   :long "help")  (:name :read   :description "Read serialized records file"   :short #\r   :long "read"   :arg-parser #'file-test)  (:name :print-month   :description "Print records for given month"   :short #\p   :long "print-month"   :arg-parser #'check-month)  (:name :interactive-mode   :description "Run in interactive mode"   :short #\i   :long "interactive"));; See: https://github.com/libre-man/unix-opts/blob/master/example/example.lisp(defmacro when-option ((options opt) &body body)  `(let ((it (getf ,options ,opt)))     (when it       ,@body)))(defun reload ()  (load "~/Repos/fin-lisp/fin-lisp.lisp"))(defun wfile (file-content file-path)  (alexandria:write-string-into-file   (concatenate 'string file-content) file-path :if-exists :overwrite						:if-does-not-exist :create));;;;;;;;;;;;;;;;;;;;;;;;;;; Encryption stuff ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; See: https://www.cliki.net/Ironclad;;; Return cipher when provided key(defun get-cipher (key)  (ironclad:make-cipher   :blowfish   :mode :ecb   :key (ironclad:ascii-string-to-byte-array key)));;; First serialize the file,;;; then encrypt it from disk(defun encrypt-records (key filename)  (let ((cipher (get-cipher key))	(file-content (uiop:read-file-string filename)))    (let ((content (ironclad:ascii-string-to-byte-array file-content)))      (ironclad:encrypt-in-place cipher content)      (wfile       (write-to-string (ironclad:octets-to-integer content))       (concatenate 'string filename ".enc")))))(defun decrypt-records (key filename)  (let ((cipher (get-cipher key))	(file-content (uiop:read-file-string filename)))    (let ((content (ironclad:integer-to-octets (parse-integer file-content))))      (ironclad:decrypt-in-place cipher content)      (coerce (mapcar #'code-char (coerce content 'list)) 'string))));;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; End Encryption Stuff ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;(defun reset-records ()  (setf *records* (make-hash-table :test 'equal)))  ;; Called like: (add-month '202107)(defun add-month (month-key)  (setf (gethash month-key *records*) (make-hash-table :test 'equalp))  month-key);;; Taken from practical common lisp(defun prompt-read (prompt)  (format *query-io* "~a: " prompt)  (force-output *query-io*)  (read-line *query-io*))(defun prompt-for-expense ()  (list   (prompt-read "Enter expense name")   (parse-integer    (prompt-read "Enter expense value"))))(defun add-expense-to-month (month)  (if (gethash month *records*)      (let ((innerhash (gethash month *records*))	    (exp-l (prompt-for-expense)))	(setf (gethash (first exp-l) innerhash) (second exp-l)))      ;;NIL))      (add-expense-to-month (add-month month))));;; Given key for *records* hash,;;; print expenses/values for month(defun dump-month (month-key)  (format t "~a~C" month-key #\linefeed)  (let ((month-hash)	(exp-keys))    (setf month-hash (gethash month-key *records*))    (setf exp-keys (loop for key being the hash-keys of month-hash collect key))    (dolist (exp-key exp-keys)      (format t "~a : ~a~C" exp-key (gethash exp-key month-hash) #\linefeed))));;; Dump all records.(defun dump-records ()  (let ((record-key-list (loop for key being the hash-keys of *records* collect key)))    (dolist (month-key record-key-list) (dump-month month-key))))(defun serialize-records (key filename)  (with-open-file (stream filename			  :direction :output			  :if-exists :overwrite			  :if-does-not-exist :create)    (yason:encode *records* stream))  (encrypt-records key filename))(defun deserialize-records (key filename)  ;; Import records from old perl version (plaintext file)(defun import-records (filename)  (let ((old-file-lines	  (with-open-file (stream filename)	    (loop for line = (read-line stream nil)		  while line		  collect line)))	(mre (ppcre:create-scanner "^(.*)[0-9]{4}$"))	(ere (ppcre:create-scanner "^([A-Z].*)\ -\ \\\$([0-9]{1,4}) - PAID"))	(cur-mon)	(cur-exp))    (loop for line in old-file-lines	  do (progn	       (if (ppcre:scan mre line) (setf cur-mon line))	       (if (ppcre:scan ere line)		   (progn		     (setf cur-exp (ppcre:register-groups-bind (first second) (ere line) :sharedp t (list first second)))		     (print cur-exp)		     (if (gethash cur-mon *records*)			 (let ((innerhash (gethash cur-mon *records*)))			   (setf (gethash (first cur-exp) innerhash) (second cur-exp))))		     (if (not (gethash cur-mon *records*))			 (progn			   (add-month cur-mon)			   (let ((innerhash (gethash cur-mon *records*)))			     (setf (gethash (first cur-exp) innerhash) (second cur-exp))))))))))) (defmacro generic-handler (form error-string)  `(handler-case ,form    (error (e)      (format t "Invalid input: ~a ~%" ,error-string)      (values 0 e))));; Util screen clearer(defun cls()  (format t "~A[H~@*~A[J" #\escape))(defun interactive-mode ()  (format t "~%")  (format t "Available options:~%")  (format t "1. Enter expense~%")  (format t "2. Display month~%")  (format t "3. Write records~%")  (format t "4. Read records~%")  (format t "5. Quit~%")  (format t "6. Import Records~%")  (let      ((answer (prompt-read "Select an option")))    (if (string= answer "1")	(generic-handler	 (add-expense-to-month (prompt-read "Enter month"))	 "Invalid Input"))    (if (string= answer "2")	(generic-handler	 (dump-month (prompt-read "Enter month"))	 "Invalid month"))    (if (string= answer "3")	(generic-handler	 (serialize-records (prompt-read "Enter encryption key")			    (prompt-read "Enter filename"))	 "Serialization error or invalid filename"))    (if (string= answer "4")	(generic-handler	 (deserialize-records (prompt-read "Enter filename"))	 "Deserialization error or invalid filename"))    (if (string= answer "5")	(quit))    (if (string= answer "6")	(generic-handler	 (import-records (prompt-read "Enter filename"))	 "Parsing error or invalid filename")))  (interactive-mode))(defun display-help ()  (format t "foo ~%")  (opts:describe   :prefix "fin-lisp.lisp - Basic expense tracker in lisp"   :usage-of "fin-lisp.lisp"   :args "[FREE-ARGS]")  (quit));; Entry point(defun main ()  (if (= 1 (length sb-ext:*posix-argv*)) (interactive-mode))  (let ((matches (opts:get-opts)))    (format t "~a ~%" matches)    (when-option (matches :help)      (display-help))    (when-option (matches :print-month)          (when-option (matches :interactive-mode)      (progn	(interactive-mode)	(quit))))))		 
 |