Add connection registration

This commit is contained in:
Simon Ser 2020-02-04 18:56:07 +01:00
parent 23ca41b435
commit 5547eb7290
No known key found for this signature in database
GPG Key ID: 0FDE7BE0E88F5E48
2 changed files with 77 additions and 8 deletions

View File

@ -15,7 +15,7 @@ func main() {
log.Fatalf("failed to start listener: %v", err) log.Fatalf("failed to start listener: %v", err)
} }
s := jounce.Server{} s := jounce.Server{Hostname: "localhost"}
log.Printf("Server listening on %v", addr) log.Printf("Server listening on %v", addr)
log.Fatal(s.Serve(ln)) log.Fatal(s.Serve(ln))

View File

@ -42,12 +42,27 @@ func (err ircError) Error() string {
type conn struct { type conn struct {
net net.Conn net net.Conn
irc *irc.Conn irc *irc.Conn
srv *Server
registered bool registered bool
closed bool
nick string nick string
username string username string
realname string realname string
} }
func (c *conn) Close() error {
if err := c.net.Close(); err != nil {
return err
}
c.closed = true
return nil
}
func (c *conn) WriteMessage(msg *irc.Message) error {
msg.Prefix = c.srv.prefix()
return c.irc.WriteMessage(msg)
}
func (c *conn) handleMessageUnregistered(msg *irc.Message) error { func (c *conn) handleMessageUnregistered(msg *irc.Message) error {
switch msg.Command { switch msg.Command {
case "NICK": case "NICK":
@ -61,10 +76,52 @@ func (c *conn) handleMessageUnregistered(msg *irc.Message) error {
} }
c.username = "~" + msg.Params[0] c.username = "~" + msg.Params[0]
c.realname = msg.Params[3] c.realname = msg.Params[3]
c.registered = true case "QUIT":
return c.Close()
default: default:
return newUnknownCommandError(msg.Command) return newUnknownCommandError(msg.Command)
} }
if c.username != "" && c.nick != "" {
return c.register()
}
return nil
}
func (c *conn) register() error {
c.registered = true
err := c.WriteMessage(&irc.Message{
Command: irc.RPL_WELCOME,
Params: []string{c.nick, "Welcome to jounce, " + c.nick},
})
if err != nil {
return err
}
err = c.WriteMessage(&irc.Message{
Command: irc.RPL_YOURHOST,
Params: []string{c.nick, "Your host is " + c.srv.Hostname},
})
if err != nil {
return err
}
err = c.WriteMessage(&irc.Message{
Command: irc.RPL_CREATED,
Params: []string{c.nick, "This server was created <datetime>"}, // TODO
})
if err != nil {
return err
}
err = c.WriteMessage(&irc.Message{
Command: irc.RPL_MYINFO,
Params: []string{c.nick, c.srv.Hostname, "unknown", "", ""},
})
if err != nil {
return err
}
return nil return nil
} }
@ -78,17 +135,24 @@ func (c *conn) handleMessage(msg *irc.Message) error {
"You may not reregister", "You may not reregister",
}, },
}} }}
case "QUIT":
return c.Close()
default: default:
return newUnknownCommandError(msg.Command) return newUnknownCommandError(msg.Command)
} }
} }
type Server struct{} type Server struct{
Hostname string
}
func (s *Server) prefix() *irc.Prefix {
return &irc.Prefix{Name: s.Hostname}
}
func (s *Server) handleConn(netConn net.Conn) error { func (s *Server) handleConn(netConn net.Conn) error {
defer netConn.Close() c := conn{net: netConn, irc: irc.NewConn(netConn), srv: s}
defer c.Close()
c := conn{net: netConn, irc: irc.NewConn(netConn)}
for { for {
msg, err := c.irc.ReadMessage() msg, err := c.irc.ReadMessage()
if err == io.EOF { if err == io.EOF {
@ -104,15 +168,20 @@ func (s *Server) handleConn(netConn net.Conn) error {
err = c.handleMessageUnregistered(msg) err = c.handleMessageUnregistered(msg)
} }
if ircErr, ok := err.(ircError); ok { if ircErr, ok := err.(ircError); ok {
if err := c.irc.WriteMessage(ircErr.Message); err != nil { ircErr.Message.Prefix = s.prefix()
if err := c.WriteMessage(ircErr.Message); err != nil {
return fmt.Errorf("failed to write IRC reply: %v", err) return fmt.Errorf("failed to write IRC reply: %v", err)
} }
} else if err != nil { } else if err != nil {
return fmt.Errorf("failed to handle IRC command %q: %v", msg.Command, err) return fmt.Errorf("failed to handle IRC command %q: %v", msg.Command, err)
} }
if c.closed {
return nil
}
} }
return netConn.Close() return c.Close()
} }
func (s *Server) Serve(ln net.Listener) error { func (s *Server) Serve(ln net.Listener) error {