272 lines
5.8 KiB
Go
272 lines
5.8 KiB
Go
// vi:ts=4:sts=4:sw=4:noet:tw=72
|
|
|
|
package modules
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"math/rand"
|
|
"os"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"git.dnix.de/an/flokatilib/util"
|
|
"git.dnix.de/an/xlog"
|
|
)
|
|
|
|
type quizQuestion struct {
|
|
category, question, answer, regexp, level string
|
|
}
|
|
|
|
var (
|
|
quizCtrlCh = make(chan string, 1024)
|
|
quizAnswerCh = make(chan *Message, 1024)
|
|
quizQuestions []quizQuestion
|
|
quizQuestionValues = map[string]int{"extreme": 5, "hard": 4, "normal": 3, "easy": 2, "baby": 1}
|
|
)
|
|
|
|
func init() {
|
|
MsgFuncs["quiz"] = quizHandleMessage
|
|
RunFuncs["quiz"] = quiz
|
|
rand.Seed(time.Now().Unix())
|
|
}
|
|
|
|
func quizHandleMessage(m *Message) {
|
|
tok := strings.Split(m.Text, " ")
|
|
if len(tok) < 1 {
|
|
return
|
|
}
|
|
switch tok[0] {
|
|
case "!quiz", "!quizstart":
|
|
quizCtrlCh <- "start"
|
|
break
|
|
case "!quizstop":
|
|
quizCtrlCh <- "stop"
|
|
break
|
|
default:
|
|
quizAnswerCh <- m
|
|
break
|
|
}
|
|
}
|
|
|
|
func quiz() {
|
|
time.Sleep(5 * time.Second)
|
|
SayCh <- fmt.Sprintf("%s\nquiz mod test. !quizstart to start, !quizstop to end.", "*")
|
|
for {
|
|
time.Sleep(1 * time.Millisecond)
|
|
select {
|
|
case s := <-quizCtrlCh:
|
|
if s == "start" {
|
|
quizRun()
|
|
}
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
func quizRun() {
|
|
quizQuestions = quizLoadQuestions("questions.txt")
|
|
ranklist := make(map[string]int)
|
|
SayCh <- fmt.Sprintf("%s\nQuiz gestartet.", "*")
|
|
for {
|
|
solved, cont, solver, gain := quizAskQuestion()
|
|
if !cont {
|
|
SayCh <- fmt.Sprintf("%s\nQuiz beendet.", "*")
|
|
return
|
|
} else {
|
|
if solved {
|
|
if _, exists := ranklist[solver]; exists {
|
|
ranklist[solver] += gain
|
|
} else {
|
|
ranklist[solver] = gain
|
|
}
|
|
if ranklist[solver] > 1000 {
|
|
bold := byte(0x02)
|
|
solver = string(bold) + solver + string(bold)
|
|
SayCh <- fmt.Sprintf("%s\n%s gewinnt!", "*", solver)
|
|
quizPrintRanklist(ranklist)
|
|
SayCh <- fmt.Sprintf("%s\nQuiz beendet.", "*")
|
|
return
|
|
}
|
|
}
|
|
}
|
|
quizPrintRanklist(ranklist)
|
|
}
|
|
}
|
|
|
|
func quizPrintRanklist(ranklist map[string]int) {
|
|
if len(ranklist) == 0 {
|
|
return
|
|
}
|
|
ranklistc := make(map[string]int, 0)
|
|
for k, v := range ranklist {
|
|
ranklistc[k] = v
|
|
}
|
|
SayCh <- fmt.Sprintf("%s\nAktueller Punktestand:", "*")
|
|
for {
|
|
maxk := ""
|
|
maxv := -1
|
|
if len(ranklistc) == 0 {
|
|
break
|
|
}
|
|
for k, v := range ranklistc {
|
|
if v > maxv {
|
|
maxv = v
|
|
maxk = k
|
|
}
|
|
}
|
|
delete(ranklistc, maxk)
|
|
SayCh <- fmt.Sprintf("%s\n%s: %d", "*", maxk, maxv)
|
|
}
|
|
}
|
|
|
|
func quizAskQuestion() (bool, bool, string, int) {
|
|
q := quizQuestions[rand.Intn(len(quizQuestions))]
|
|
SayCh <- fmt.Sprintf("%s\nEine Frage aus der Kategorie \"%s\" (%s):", "*", q.category, q.level)
|
|
SayCh <- fmt.Sprintf("%s\n>>> %s <<<", "*", q.question)
|
|
solved, cont, solver, gain := quizWaitForAnswer(q)
|
|
if !solved {
|
|
SayCh <- fmt.Sprintf("%s\nDie richtige Antwort wäre gewesen:", "*")
|
|
SayCh <- fmt.Sprintf("%s\n%s [%s]", "*", q.answer, q.regexp)
|
|
}
|
|
time.Sleep(5 * time.Second)
|
|
return solved, cont, solver, gain
|
|
}
|
|
|
|
func quizWaitForAnswer(q quizQuestion) (bool, bool, string, int) {
|
|
i := 0
|
|
haveAnswer := false
|
|
timer := time.Now().Unix()
|
|
timeHintCh := make(chan bool)
|
|
go func(chan bool) {
|
|
for i := 0; i < 4; i++ {
|
|
time.Sleep(15 * time.Second)
|
|
timeHintCh <- true
|
|
}
|
|
}(timeHintCh)
|
|
for {
|
|
select {
|
|
case m := <-quizAnswerCh:
|
|
haveAnswer = true
|
|
points := 10 - ((time.Now().Unix() - timer) / 10)
|
|
if points < 1 {
|
|
points = 1
|
|
}
|
|
re, err := regexp.Compile(q.regexp)
|
|
if err != nil {
|
|
xlog.Error(err.Error())
|
|
return false, false, "", 0
|
|
}
|
|
if re.MatchString(strings.ToLower(util.ReplaceUmlauts(m.Text))) {
|
|
bold := byte(0x02)
|
|
solver := m.From
|
|
solver = string(bold) + solver + string(bold)
|
|
value := quizQuestionValues[q.level]
|
|
gain := int(points) * value
|
|
SayCh <- fmt.Sprintf("%s\n%s hat die Frage korrekt beantwortet und erhält dafür %d (%d * %d) Punkte.", "*", solver, gain, points, value)
|
|
return true, true, m.From, gain
|
|
}
|
|
break
|
|
case s := <-quizCtrlCh:
|
|
if s == "stop" {
|
|
return false, false, "", 0
|
|
}
|
|
break
|
|
case <-timeHintCh:
|
|
i++
|
|
if i > 3 {
|
|
if haveAnswer {
|
|
SayCh <- fmt.Sprintf("%s\nNiemand konnte die Frage korrekt beantworten.", "*")
|
|
return false, true, "", 0
|
|
} else {
|
|
SayCh <- fmt.Sprintf("%s\nNiemand hat auf die Frage geantwortet.", "*")
|
|
return false, false, "", 0
|
|
}
|
|
} else {
|
|
quizGiveHint(q, i)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func quizGiveHint(q quizQuestion, n int) {
|
|
var hint string
|
|
haveHint := false
|
|
for {
|
|
hint = ""
|
|
for i := 0; i < len(q.answer); i++ {
|
|
if string(q.answer[i]) == " " {
|
|
hint += " "
|
|
} else {
|
|
if rand.Intn(10) < 2 {
|
|
haveHint = true
|
|
hint += string(q.answer[i])
|
|
} else {
|
|
hint += "-"
|
|
}
|
|
}
|
|
}
|
|
if haveHint {
|
|
break
|
|
}
|
|
}
|
|
SayCh <- fmt.Sprintf("%s\nTipp: %s", "*", hint)
|
|
}
|
|
|
|
func quizLoadQuestions(path string) []quizQuestion {
|
|
file, err := os.Open(path)
|
|
if err != nil {
|
|
xlog.Fatal(err.Error())
|
|
}
|
|
defer file.Close()
|
|
|
|
questions := make([]quizQuestion, 0)
|
|
q := quizQuestion{"", "", "", "", ""}
|
|
|
|
scanner := bufio.NewScanner(file)
|
|
for scanner.Scan() {
|
|
line := scanner.Text()
|
|
if len(line) == 0 || line[0] == '#' || line[0] == '\n' {
|
|
if q.category != "" {
|
|
questions = append(questions, q)
|
|
q = quizQuestion{"", "", "", "", "normal"}
|
|
}
|
|
} else {
|
|
tok := strings.Split(line, ":: ")
|
|
switch tok[0] {
|
|
case "Category":
|
|
q.category = tok[1]
|
|
break
|
|
case "Question":
|
|
q.question = tok[1]
|
|
break
|
|
case "Answer":
|
|
q.answer = strings.Replace(util.ReplaceUmlauts(tok[1]), "#", "", -1)
|
|
if q.regexp == "" {
|
|
regexp := strings.Split(strings.ToLower(util.ReplaceUmlauts(tok[1])), "#")
|
|
if len(regexp) > 1 {
|
|
q.regexp = regexp[1]
|
|
} else {
|
|
q.regexp = regexp[0]
|
|
}
|
|
}
|
|
break
|
|
case "Regexp":
|
|
q.regexp = util.ReplaceUmlauts(strings.ToLower(tok[1]))
|
|
break
|
|
case "Level":
|
|
q.level = tok[1]
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
xlog.Fatal(err.Error())
|
|
}
|
|
|
|
return questions
|
|
}
|