Eliza Program - Newer Syntax

;;; SHRINK.LSP -- a simple conversational program after ELIZA.

;;; (C) Copyright 1995 by Steven L. Tanimoto.
;;; This program is described in Chapter 3 ("Productions Systems
;;; and Pattern Matching") of
;;; "The Elements of Artificial Intelligence Using Common Lisp," 2nd ed.,
;;; published by W. H. Freeman, 41 Madison Ave., New York, NY 10010.
;;; Permission is granted for noncommercial use and modification of
;;; this program, provided that this copyright notice is retained
;;; and followed by a notice of any modifications made to the program.

;;; Requires the functions MATCH and VAL defined in MATCH.CL.

(defun shrink ()
  "SHRINK is the top-level function."
  (let ((wword-count 0) ; counter for question words.
        (punt-count 0)  ; counter for punt responses.
        b  ; holds bindings after matching.
        s  ; holds the mapped input sentence.
        )
  (format t "WELCOME TO MY SOFA!~%")
  (format t "PLEASE ENCLOSE YOUR INPUT IN PARENTHESES.~%")
  (loop (setf s (you-me-map (read)))
        (terpri)
        (cond ((match '(bye) s)
               (return 'goodbye) )
              ((setf b (match '(you are (* x)) s))
               (incf wword-count)
               (print-s (append '(please tell me)
                                (list (wword wword-count))
                                '(you are)
                                (val 'x b) )) )
              ((setf b (match '(you have (* x)) s))
               (print-q (append '(how long have you had) (val 'x b) )) )
              ((match '(you feel (* x)) s)
               (format t "I SOMETIMES FEEL THE SAME WAY.~%") )
              ((match '(because (* x)) s)
               (format t "IS THAT REALLY THE REASON.~%") )
              ((match nil s)
               (format t "PLEASE SAY SOMETHING!~%") )
              ((setf b (match '(yes (* x)) s))
               (print-q (append '(how can you be so sure) (val 'x b) )) )
              ((setf b (match '(me are (* x)) s))
               (print-s (append '(oh yeah i am) (val 'x b) )) )
              ((setf b (match '((verbp v) (* x)) s))
               (print-q (append '(why do you want me to)
                               (list (val 'v b)) (val 'x b) ) ) )
              ((setf b (match '((wpred w)(* x)) s))
               (print-s (append '(you tell me)(list (val 'w b))) ) )
              ((match '(do me think (* x)) s)
               (format t "I THINK YOU SHOULD ANSWER THAT YOURSELF.~%") )
              ((setf b (match '((dpred w) me (* x)) s))
               (print-s (append '(perhaps i)(list (val 'w b)) (val 'x b) ) ) )
              ((member 'dream s)
               (format t "FOR DREAM ANALYSIS SEE FREUD.~%") )
              ((member 'love s)
               (format t "ALL IS FAIR IN LOVE AND WAR.~%") )
              ((member 'no s)
               (format t "DONT BE SO NEGATIVE.~%") )
              ((member 'maybe s)
               (format t "BE MORE DECISIVE!~%") )
              ((member 'you s)(print-s s))
              (t (incf punt-count)
                 (print-s (punt punt-count)) ) ) ) ) )

(defun print-s (message)
  "Print message list as a sentence."
  (printl (butlast message))
  ; Print the last element followed by a period:
  (format t "~a." (first (last message)))
  (terpri) )

(defun print-q (message)
  "Print message list as a question."
  (printl (butlast message))
  ; Print the last element followed by a question mark:
  (format t "~a?" (first (last message)))
  (terpri) )

(defun printl (message)
  "Prints a list without the surrounding parens."
  ; Print each element except the last followed by a space:
  (mapcar #'(lambda (txt) (format t "~a " txt))
          message) )

;;; WWORD returns one of four question words,
;;; WWORD-COUNT must be an integer.
(defun wword (wword-count)
  "Returns the ith question word where i = WWORD-COUNT mod 4."
  (nth (mod wword-count 4)
          '(when why where how) ) )

(defun wpred (w)
  "Returns T if W is one of the question words."
  (member w '(why where when what which how)) )

(defun dpred (w)
  "Returns T is W is an auxiliary verb."
  (member w '(do can should would)) )

(defun punt (punt-count)
  "Returns one from a list of default responses."
  (nth (mod punt-count 6)
            '((please go on)
              (tell me more)
              (i see)
              (what does that indicate)
              (but why be concerned about it)
              (just tell me how you feel) ) ) )

(defun you-me (w)
  "Changes a word from 1st to 2nd person or vice-versa."
  (cond
    ((eql w 'i) 'you) ((eql w 'me) 'you) ((eql w 'you) 'me)
    ((eql w 'my) 'your) ((eql w 'your) 'my) 
    ((eql w 'yours) 'mine) ((eql w 'mine) 'yours)
    ((eql w 'am) 'are) (t w) ) )

(defun you-me-map (lst)
  "Applies YOU-ME to a whole sentence or phrase."
  (mapcar (function you-me) lst) )

(defun verbp (w)
  "Return T if W is one of these known verbs."
  (member w '(go have be try eat take help make get jump
              write type fill put turn compute
              think drink blink crash crunch add) ) )