quiz.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. package models
  2. import (
  3. "crypto/sha256"
  4. "errors"
  5. "fmt"
  6. "io"
  7. "sort"
  8. "strings"
  9. "gopkg.in/yaml.v2"
  10. )
  11. type Quiz struct {
  12. Meta
  13. Hash string `json:"hash"`
  14. Question *Question `json:"question" gorm:"foreignKey:ID"`
  15. Answers []*Answer `json:"answers" gorm:"many2many:quiz_answers"`
  16. Tags []*Tag `json:"tags" yaml:"-" gorm:"-"`
  17. Correct *Answer `json:"correct" gorm:"foreignKey:ID"`
  18. CorrectPos uint `gorm:"-"` // Position of the correct answer during quiz creation
  19. Type int `json:"type"`
  20. }
  21. func MarkdownToQuiz(quiz *Quiz, markdown string) error {
  22. meta, remainingMarkdown, err := ParseMetaHeaderFromMarkdown(markdown)
  23. if err != nil {
  24. return err
  25. }
  26. lines := strings.Split(remainingMarkdown, "\n")
  27. questionText := ""
  28. answers := []*Answer{}
  29. for _, line := range lines {
  30. if strings.HasPrefix(line, "*") {
  31. answerText := strings.TrimPrefix(line, "* ")
  32. answer := &Answer{Text: answerText}
  33. answers = append(answers, answer)
  34. } else {
  35. if questionText != "" {
  36. questionText += "\n"
  37. }
  38. questionText += line
  39. }
  40. }
  41. questionText = strings.TrimRight(questionText, "\n")
  42. if questionText == "" {
  43. return fmt.Errorf("Question text should not be empty.")
  44. }
  45. if len(answers) < 2 {
  46. return fmt.Errorf("Number of answers should be at least 2 but parsed answers are %d.", len(answers))
  47. }
  48. question := &Question{Text: questionText}
  49. quiz.Question = question
  50. quiz.Answers = answers
  51. //quiz = &Quiz{Question: question, Answers: answers, CorrectPos: 0}
  52. if meta != nil {
  53. quiz.Meta = *meta
  54. }
  55. return nil
  56. }
  57. func QuizToMarkdown(quiz *Quiz) (string, error) {
  58. if quiz.Question == nil {
  59. return "", errors.New("Quiz should contain a question but it wasn't provided.")
  60. }
  61. if len(quiz.Answers) < 2 {
  62. return "", errors.New("Quiz should contain at least 2 answers but none was provided.")
  63. }
  64. quiz.Correct = quiz.Answers[quiz.CorrectPos]
  65. if quiz.Correct == nil {
  66. return "", errors.New("Quiz should contain a correct answer but not was provided.")
  67. }
  68. correctAnswer := "* " + quiz.Correct.Text
  69. var otherAnswers string
  70. for pos, answer := range quiz.Answers {
  71. if quiz.CorrectPos != uint(pos) {
  72. otherAnswers += "* " + answer.Text + "\n"
  73. }
  74. }
  75. markdown := quiz.Question.Text + "\n\n" + correctAnswer + "\n" + otherAnswers
  76. return markdown, nil
  77. }
  78. func (q *Quiz) GetHash() string {
  79. return q.calculateHash()
  80. }
  81. func (q *Quiz) Marshal() ([]byte, error) {
  82. result, err := QuizToMarkdown(q)
  83. return []byte(result), err
  84. }
  85. func (q *Quiz) Unmarshal(data []byte) error {
  86. return MarkdownToQuiz(q, string(data))
  87. }
  88. func (q *Quiz) calculateHash() string {
  89. result := make([]string, 0)
  90. result = append(result, q.Question.GetHash())
  91. for _, a := range q.Answers {
  92. result = append(result, a.GetHash())
  93. }
  94. orderedHashes := make([]string, len(result))
  95. copy(orderedHashes, result)
  96. sort.Strings(orderedHashes)
  97. return fmt.Sprintf("%x", sha256.Sum256([]byte(strings.Join(orderedHashes, ""))))
  98. }
  99. func ParseMetaHeaderFromMarkdown(markdown string) (*Meta, string, error) {
  100. reader := strings.NewReader(markdown)
  101. var sb strings.Builder
  102. var line string
  103. var err error
  104. for {
  105. line, err = readLine(reader)
  106. if err != nil {
  107. if err == io.EOF {
  108. break
  109. }
  110. return nil, "", err
  111. }
  112. if strings.TrimSpace(line) == "---" {
  113. break
  114. }
  115. }
  116. for {
  117. line, err = readLine(reader)
  118. if err != nil {
  119. if err == io.EOF {
  120. break
  121. }
  122. return nil, "", err
  123. }
  124. if strings.TrimSpace(line) == "---" {
  125. break
  126. }
  127. sb.WriteString(line)
  128. }
  129. if sb.String() == "" {
  130. return nil, markdown, nil
  131. }
  132. var meta Meta
  133. err = yaml.Unmarshal([]byte(sb.String()), &meta)
  134. if err != nil {
  135. return nil, markdown, err
  136. }
  137. remainingMarkdown := markdown[strings.Index(markdown, "---\n"+sb.String()+"---\n")+len("---\n"+sb.String()+"---\n"):]
  138. return &meta, remainingMarkdown, nil
  139. }
  140. func readLine(reader *strings.Reader) (string, error) {
  141. var sb strings.Builder
  142. for {
  143. r, _, err := reader.ReadRune()
  144. if err != nil {
  145. if err == io.EOF {
  146. return sb.String(), io.EOF
  147. }
  148. return "", err
  149. }
  150. sb.WriteRune(r)
  151. if r == '\n' {
  152. break
  153. }
  154. }
  155. return sb.String(), nil
  156. }