package main import ( "bufio" "fmt" "log" "net" "os" "strconv" ) var ( gConnList = make(map[int]net.Conn) gQuitChan = make(chan struct{}, 1) gListener net.Listener ) func serve() { f, err := os.Create(gServerLogPath) if err != nil { panic(err) } defer f.Close() log.SetOutput(f) log.Print("hi!") l, err := net.Listen(gSocketProt, gSocketPath) if err != nil { log.Printf("listening socket: %s", err) return } defer l.Close() gListener = l listen(l) } func listen(l net.Listener) { for { c, err := l.Accept() if err != nil { select { case <-gQuitChan: log.Printf("bye!") return default: log.Printf("accepting connection: %s", err) } } go handleConn(c) } } func handleConn(c net.Conn) { s := bufio.NewScanner(c) Loop: for s.Scan() { log.Printf("listen: %s", s.Text()) word, rest := splitWord(s.Text()) switch word { case "conn": if rest != "" { word2, _ := splitWord(rest) id, err := strconv.Atoi(word2) if err != nil { log.Print("listen: conn: client id should be a number") } else { gConnList[id] = c } } else { log.Print("listen: conn: requires a client id") } case "drop": if rest != "" { word2, _ := splitWord(rest) id, err := strconv.Atoi(word2) if err != nil { log.Print("listen: drop: client id should be a number") } else { delete(gConnList, id) } } else { log.Print("listen: drop: requires a client id") } case "send": if rest != "" { word2, rest2 := splitWord(rest) id, err := strconv.Atoi(word2) if err != nil { for _, c := range gConnList { fmt.Fprintln(c, rest) } } else { if c, ok := gConnList[id]; ok { fmt.Fprintln(c, rest2) } } } case "quit": gQuitChan <- struct{}{} for _, c := range gConnList { fmt.Fprintln(c, "echo server is quitting...") c.Close() } gListener.Close() break Loop default: log.Printf("listen: unexpected command: %s", word) } } if s.Err() != nil { log.Printf("listening: %s", s.Err()) } c.Close() }