2016-08-13 12:49:04 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"os"
|
2016-11-06 14:06:25 +00:00
|
|
|
"strconv"
|
2016-08-13 12:49:04 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2018-03-22 14:54:24 +00:00
|
|
|
gConnList = make(map[int]net.Conn)
|
2020-12-27 01:23:03 +00:00
|
|
|
gQuitChan = make(chan struct{}, 1)
|
2018-07-22 17:44:56 +00:00
|
|
|
gListener net.Listener
|
2016-08-13 12:49:04 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func serve() {
|
2022-02-06 16:43:01 +00:00
|
|
|
if gLogPath != "" {
|
|
|
|
f, err := os.OpenFile(gLogPath, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
log.SetOutput(f)
|
2022-01-22 10:47:18 +00:00
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
|
|
|
|
log.Print("hi!")
|
|
|
|
|
2022-01-26 13:52:40 +00:00
|
|
|
if gSocketProt == "unix" {
|
|
|
|
setUserUmask()
|
|
|
|
}
|
|
|
|
|
2017-07-20 22:26:36 +00:00
|
|
|
l, err := net.Listen(gSocketProt, gSocketPath)
|
2016-08-13 12:49:04 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("listening socket: %s", err)
|
2016-11-06 14:06:25 +00:00
|
|
|
return
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
defer l.Close()
|
|
|
|
|
2018-07-22 17:44:56 +00:00
|
|
|
gListener = l
|
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
listen(l)
|
|
|
|
}
|
|
|
|
|
|
|
|
func listen(l net.Listener) {
|
|
|
|
for {
|
|
|
|
c, err := l.Accept()
|
|
|
|
if err != nil {
|
2018-07-22 17:44:56 +00:00
|
|
|
select {
|
|
|
|
case <-gQuitChan:
|
|
|
|
log.Printf("bye!")
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
log.Printf("accepting connection: %s", err)
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
2016-10-29 23:20:35 +00:00
|
|
|
go handleConn(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-21 13:54:01 +00:00
|
|
|
func echoerr(c net.Conn, msg string) {
|
|
|
|
fmt.Fprintln(c, msg)
|
|
|
|
log.Printf(msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func echoerrf(c net.Conn, format string, a ...interface{}) {
|
|
|
|
echoerr(c, fmt.Sprintf(format, a...))
|
|
|
|
}
|
|
|
|
|
2016-10-29 23:20:35 +00:00
|
|
|
func handleConn(c net.Conn) {
|
|
|
|
s := bufio.NewScanner(c)
|
|
|
|
|
2019-08-12 11:52:27 +00:00
|
|
|
Loop:
|
2016-10-29 23:20:35 +00:00
|
|
|
for s.Scan() {
|
2016-11-06 14:06:25 +00:00
|
|
|
log.Printf("listen: %s", s.Text())
|
|
|
|
word, rest := splitWord(s.Text())
|
|
|
|
switch word {
|
2016-10-29 23:20:35 +00:00
|
|
|
case "conn":
|
2016-11-06 14:06:25 +00:00
|
|
|
if rest != "" {
|
|
|
|
word2, _ := splitWord(rest)
|
|
|
|
id, err := strconv.Atoi(word2)
|
|
|
|
if err != nil {
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerr(c, "listen: conn: client id should be a number")
|
2016-11-06 14:06:25 +00:00
|
|
|
} else {
|
|
|
|
gConnList[id] = c
|
|
|
|
}
|
|
|
|
} else {
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerr(c, "listen: conn: requires a client id")
|
2016-11-06 14:06:25 +00:00
|
|
|
}
|
2021-05-19 18:45:27 +00:00
|
|
|
case "drop":
|
|
|
|
if rest != "" {
|
|
|
|
word2, _ := splitWord(rest)
|
|
|
|
id, err := strconv.Atoi(word2)
|
|
|
|
if err != nil {
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerr(c, "listen: drop: client id should be a number")
|
2021-05-19 18:45:27 +00:00
|
|
|
} else {
|
|
|
|
delete(gConnList, id)
|
|
|
|
}
|
|
|
|
} else {
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerr(c, "listen: drop: requires a client id")
|
2021-05-19 18:45:27 +00:00
|
|
|
}
|
2016-11-06 14:06:25 +00:00
|
|
|
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 {
|
2021-06-12 21:27:43 +00:00
|
|
|
if c2, ok := gConnList[id]; ok {
|
|
|
|
fmt.Fprintln(c2, rest2)
|
|
|
|
} else {
|
|
|
|
echoerr(c, "listen: send: no such client id is connected")
|
2016-11-06 14:06:25 +00:00
|
|
|
}
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
2021-05-19 18:47:53 +00:00
|
|
|
case "quit":
|
|
|
|
if len(gConnList) == 0 {
|
|
|
|
gQuitChan <- struct{}{}
|
|
|
|
gListener.Close()
|
|
|
|
break Loop
|
|
|
|
}
|
2021-05-19 18:47:08 +00:00
|
|
|
case "quit!":
|
2020-12-27 01:23:03 +00:00
|
|
|
gQuitChan <- struct{}{}
|
2018-07-22 17:44:56 +00:00
|
|
|
for _, c := range gConnList {
|
|
|
|
fmt.Fprintln(c, "echo server is quitting...")
|
|
|
|
c.Close()
|
|
|
|
}
|
|
|
|
gListener.Close()
|
2019-08-12 11:52:27 +00:00
|
|
|
break Loop
|
2016-10-29 23:20:35 +00:00
|
|
|
default:
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerrf(c, "listen: unexpected command: %s", word)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
}
|
2016-10-29 23:20:35 +00:00
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
if s.Err() != nil {
|
2021-05-21 13:54:01 +00:00
|
|
|
echoerrf(c, "listening: %s", s.Err())
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
c.Close()
|
|
|
|
}
|