eliza.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. // Author: Matthew Shiel
  2. // Code adapted from https://github.com/kennysong/goeliza/
  3. package eliza
  4. import (
  5. "fmt"
  6. "math/rand"
  7. "regexp"
  8. "strings"
  9. "time"
  10. )
  11. // Greetings will return a random introductory sentence for ELIZA.
  12. func Greetings() string {
  13. return randChoice(Introductions)
  14. }
  15. // GoodbyeResponse will return a random goodbye sentence for ELIZA.
  16. func GoodbyeResponse() string {
  17. return randChoice(Goodbyes)
  18. }
  19. // ReplyTo will construct a reply for a given statement using ELIZA's rules.
  20. func ReplyTo(statement string) string {
  21. // First, preprocess the statement for more effective matching
  22. statement = preprocess(statement)
  23. // Then, we check if this is a quit statement
  24. if IsQuitStatement(statement) {
  25. return GoodbyeResponse()
  26. }
  27. // Next, we try to match the statement to a statement that ELIZA can
  28. // recognize, and construct a pre-determined, appropriate response.
  29. for pattern, responses := range Psychobabble {
  30. re := regexp.MustCompile(pattern)
  31. matches := re.FindStringSubmatch(statement)
  32. // If the statement matched any recognizable statements.
  33. if len(matches) > 0 {
  34. // If we matched a regex group in parentheses, get the first match.
  35. // The matched regex group will match a "fragment" that will form
  36. // part of the response, for added realism.
  37. var fragment string
  38. if len(matches) > 1 {
  39. fragment = reflect(matches[1])
  40. }
  41. // Choose a random appropriate response, and format it with the
  42. // fragment, if needed.
  43. response := randChoice(responses)
  44. if strings.Contains(response, "%s") {
  45. response = fmt.Sprintf(response, fragment)
  46. }
  47. return response
  48. }
  49. }
  50. // If no patterns were matched, return a default response.
  51. return randChoice(DefaultResponses)
  52. }
  53. // IsQuitStatement returns if the statement is a quit statement
  54. func IsQuitStatement(statement string) bool {
  55. statement = preprocess(statement)
  56. for _, quitResponse := range QuitResponses {
  57. if statement == quitResponse {
  58. return true
  59. }
  60. }
  61. return false
  62. }
  63. // preprocess will do some normalization on a statement for better regex matching
  64. func preprocess(statement string) string {
  65. statement = strings.TrimRight(statement, "\n.!")
  66. statement = strings.ToLower(statement)
  67. return statement
  68. }
  69. // reflect flips a few words in an input fragment (such as "I" -> "you").
  70. func reflect(fragment string) string {
  71. words := strings.Split(fragment, " ")
  72. for i, word := range words {
  73. if reflectedWord, ok := ReflectedWords[word]; ok {
  74. words[i] = reflectedWord
  75. }
  76. }
  77. return strings.Join(words, " ")
  78. }
  79. // randChoice returns a random element in an (string) array.
  80. func randChoice(list []string) string {
  81. // Added for truly random generation of numbers with seeds
  82. rand.Seed(time.Now().UnixNano())
  83. randIndex := rand.Intn(len(list))
  84. return list[randIndex]
  85. }