Merge remote-tracking branch 'origin/master' into user-mask

Conflicts:
	irc/reply.go
	irc/server.go
	irc/types.go
This commit is contained in:
Jeremy Latt 2014-03-06 17:44:37 -08:00
commit 5d46e7d7fa
7 changed files with 224 additions and 101 deletions

View File

@ -54,18 +54,26 @@ func (channel *Channel) ClientIsOperator(client *Client) bool {
return client.flags[Operator] || channel.members.HasMode(client, ChannelOperator) return client.flags[Operator] || channel.members.HasMode(client, ChannelOperator)
} }
func (channel *Channel) Nicks() []string { func (channel *Channel) Nicks(target *Client) []string {
isMultiPrefix := (target != nil) && target.capabilities[MultiPrefix]
nicks := make([]string, len(channel.members)) nicks := make([]string, len(channel.members))
i := 0 i := 0
for client, modes := range channel.members { for client, modes := range channel.members {
switch { if isMultiPrefix {
case modes[ChannelOperator]: if modes[ChannelOperator] {
nicks[i] = "@" + client.Nick() nicks[i] += "@"
case modes[Voice]:
nicks[i] = "+" + client.Nick()
default:
nicks[i] = client.Nick()
} }
if modes[Voice] {
nicks[i] += "+"
}
} else {
if modes[ChannelOperator] {
nicks[i] += "@"
} else if modes[Voice] {
nicks[i] += "+"
}
}
nicks[i] += client.Nick()
i += 1 i += 1
} }
return nicks return nicks

View File

@ -13,7 +13,10 @@ func IsNickname(nick string) bool {
type Client struct { type Client struct {
atime time.Time atime time.Time
authorized bool
awayMessage string awayMessage string
capabilities CapabilitySet
capState CapState
channels ChannelSet channels ChannelSet
commands chan editableCommand commands chan editableCommand
ctime time.Time ctime time.Time
@ -36,11 +39,13 @@ func NewClient(server *Server, conn net.Conn) *Client {
now := time.Now() now := time.Now()
client := &Client{ client := &Client{
atime: now, atime: now,
capState: CapNone,
capabilities: make(CapabilitySet),
channels: make(ChannelSet), channels: make(ChannelSet),
commands: make(chan editableCommand), commands: make(chan editableCommand),
ctime: now, ctime: now,
flags: make(map[UserMode]bool), flags: make(map[UserMode]bool),
phase: server.InitPhase(), phase: Registration,
server: server, server: server,
} }
client.socket = NewSocket(conn, client.commands) client.socket = NewSocket(conn, client.commands)
@ -68,6 +73,12 @@ func (client *Client) run() {
} }
} }
func (client *Client) connectionTimeout() {
client.commands <- &QuitCommand{
message: "connection timeout",
}
}
// //
// idle timer goroutine // idle timer goroutine
// //
@ -76,14 +87,6 @@ func (client *Client) connectionIdle() {
client.server.idle <- client client.server.idle <- client
} }
//
// quit timer goroutine
//
func (client *Client) connectionTimeout() {
client.server.timeout <- client
}
// //
// server goroutine // server goroutine
// //
@ -233,7 +236,10 @@ func (client *Client) ChangeNickname(nickname string) {
} }
} }
func (client *Client) Reply(reply string) { func (client *Client) Reply(reply string, args ...interface{}) {
if len(args) > 0 {
reply = fmt.Sprintf(reply, args...)
}
client.socket.Write(reply) client.socket.Write(reply)
} }

View File

@ -706,20 +706,34 @@ func NewOperCommand(args []string) (editableCommand, error) {
return cmd, nil return cmd, nil
} }
// TODO
type CapCommand struct { type CapCommand struct {
BaseCommand BaseCommand
args []string subCommand CapSubCommand
capabilities CapabilitySet
} }
func (msg *CapCommand) String() string { func (msg *CapCommand) String() string {
return fmt.Sprintf("CAP(args=%s)", msg.args) return fmt.Sprintf("CAP(subCommand=%s, capabilities=%s)",
msg.subCommand, msg.capabilities)
} }
func NewCapCommand(args []string) (editableCommand, error) { func NewCapCommand(args []string) (editableCommand, error) {
return &CapCommand{ if len(args) < 1 {
args: args, return nil, NotEnoughArgsError
}, nil }
cmd := &CapCommand{
subCommand: CapSubCommand(strings.ToUpper(args[0])),
capabilities: make(CapabilitySet),
}
if len(args) > 1 {
strs := spacesExpr.Split(args[1], -1)
for _, str := range strs {
cmd.capabilities[Capability(str)] = true
}
}
return cmd, nil
} }
// HAPROXY support // HAPROXY support

View File

@ -156,6 +156,7 @@ const (
ERR_TOOMANYTARGETS NumericCode = 407 ERR_TOOMANYTARGETS NumericCode = 407
ERR_NOSUCHSERVICE NumericCode = 408 ERR_NOSUCHSERVICE NumericCode = 408
ERR_NOORIGIN NumericCode = 409 ERR_NOORIGIN NumericCode = 409
ERR_INVALIDCAPCMD NumericCode = 410
ERR_NORECIPIENT NumericCode = 411 ERR_NORECIPIENT NumericCode = 411
ERR_NOTEXTTOSEND NumericCode = 412 ERR_NOTEXTTOSEND NumericCode = 412
ERR_NOTOPLEVEL NumericCode = 413 ERR_NOTOPLEVEL NumericCode = 413
@ -201,6 +202,14 @@ const (
ERR_UMODEUNKNOWNFLAG NumericCode = 501 ERR_UMODEUNKNOWNFLAG NumericCode = 501
ERR_USERSDONTMATCH NumericCode = 502 ERR_USERSDONTMATCH NumericCode = 502
CAP_LS CapSubCommand = "LS"
CAP_LIST CapSubCommand = "LIST"
CAP_REQ CapSubCommand = "REQ"
CAP_ACK CapSubCommand = "ACK"
CAP_NAK CapSubCommand = "NAK"
CAP_CLEAR CapSubCommand = "CLEAR"
CAP_END CapSubCommand = "END"
Add ModeOp = '+' Add ModeOp = '+'
List ModeOp = '=' List ModeOp = '='
Remove ModeOp = '-' Remove ModeOp = '-'
@ -231,10 +240,28 @@ const (
Secret ChannelMode = 's' // flag, deprecated Secret ChannelMode = 's' // flag, deprecated
UserLimit ChannelMode = 'l' // flag arg UserLimit ChannelMode = 'l' // flag arg
Voice ChannelMode = 'v' // arg Voice ChannelMode = 'v' // arg
MultiPrefix Capability = "multi-prefix"
SASL Capability = "sasl"
Disable CapModifier = '-'
Ack CapModifier = '~'
Sticky CapModifier = '='
)
var (
SupportedCapabilities = CapabilitySet{
MultiPrefix: true,
}
) )
const ( const (
Authorization Phase = iota
Registration Phase = iota Registration Phase = iota
Normal Phase = iota Normal Phase = iota
) )
const (
CapNone CapState = iota
CapNegotiating CapState = iota
CapNegotiated CapState = iota
)

View File

@ -250,13 +250,21 @@ func (target *Client) RplWhoReply(channel *Channel, client *Client) {
if channel != nil { if channel != nil {
channelName = channel.name channelName = channel.name
if target.capabilities[MultiPrefix] {
if channel.members[client][ChannelOperator] {
flags += "@"
}
if channel.members[client][Voice] {
flags += "+"
}
} else {
if channel.members[client][ChannelOperator] { if channel.members[client][ChannelOperator] {
flags += "@" flags += "@"
} else if channel.members[client][Voice] { } else if channel.members[client][Voice] {
flags += "+" flags += "+"
} }
} }
}
target.NumericReply(RPL_WHOREPLY, target.NumericReply(RPL_WHOREPLY,
"%s %s %s %s %s %s :%d %s", channelName, client.username, client.hostname, "%s %s %s %s %s %s :%d %s", channelName, client.username, client.hostname,
client.server.name, client.Nick(), flags, client.hops, client.realname) client.server.name, client.Nick(), flags, client.hops, client.realname)
@ -370,7 +378,7 @@ func (target *Client) RplListEnd(server *Server) {
} }
func (target *Client) RplNamReply(channel *Channel) { func (target *Client) RplNamReply(channel *Channel) {
target.MultilineReply(channel.Nicks(), RPL_NAMREPLY, target.MultilineReply(channel.Nicks(target), RPL_NAMREPLY,
"= %s :%s", channel) "= %s :%s", channel)
} }
@ -528,3 +536,8 @@ func (target *Client) ErrWasNoSuchNick(nickname string) {
target.NumericReply(ERR_WASNOSUCHNICK, target.NumericReply(ERR_WASNOSUCHNICK,
"%s :There was no such nickname", nickname) "%s :There was no such nickname", nickname)
} }
func (target *Client) ErrInvalidCapCmd(subCommand CapSubCommand) {
target.NumericReply(ERR_INVALIDCAPCMD,
"%s :Invalid CAP subcommand", subCommand)
}

View File

@ -29,7 +29,6 @@ type Server struct {
operators map[string][]byte operators map[string][]byte
password []byte password []byte
signals chan os.Signal signals chan os.Signal
timeout chan *Client
whoWas *WhoWasList whoWas *WhoWasList
} }
@ -46,7 +45,6 @@ func NewServer(config *Config) *Server {
newConns: make(chan net.Conn, 16), newConns: make(chan net.Conn, 16),
operators: config.Operators(), operators: config.Operators(),
signals: make(chan os.Signal, 1), signals: make(chan os.Signal, 1),
timeout: make(chan *Client, 16),
whoWas: NewWhoWasList(100), whoWas: NewWhoWasList(100),
} }
@ -99,14 +97,6 @@ func (server *Server) processCommand(cmd Command) {
} }
switch client.phase { switch client.phase {
case Authorization:
authCmd, ok := cmd.(AuthServerCommand)
if !ok {
client.Quit("unexpected command")
return
}
authCmd.HandleAuthServer(server)
case Registration: case Registration:
regCmd, ok := cmd.(RegServerCommand) regCmd, ok := cmd.(RegServerCommand)
if !ok { if !ok {
@ -115,7 +105,7 @@ func (server *Server) processCommand(cmd Command) {
} }
regCmd.HandleRegServer(server) regCmd.HandleRegServer(server)
default: case Normal:
srvCmd, ok := cmd.(ServerCommand) srvCmd, ok := cmd.(ServerCommand)
if !ok { if !ok {
client.ErrUnknownCommand(cmd.Code()) client.ErrUnknownCommand(cmd.Code())
@ -159,20 +149,10 @@ func (server *Server) Run() {
case client := <-server.idle: case client := <-server.idle:
client.Idle() client.Idle()
case client := <-server.timeout:
client.Quit("connection timeout")
} }
} }
} }
func (server *Server) InitPhase() Phase {
if server.password == nil {
return Registration
}
return Authorization
}
// //
// listen goroutine // listen goroutine
// //
@ -208,7 +188,7 @@ func (s *Server) listen(addr string) {
// //
func (s *Server) tryRegister(c *Client) { func (s *Server) tryRegister(c *Client) {
if c.HasNick() && c.HasUsername() { if c.HasNick() && c.HasUsername() && (c.capState != CapNegotiating) {
c.Register() c.Register()
c.RplWelcome() c.RplWelcome()
c.RplYourHost() c.RplYourHost()
@ -268,18 +248,10 @@ func (s *Server) Nick() string {
} }
// //
// authorization commands // registration commands
// //
func (msg *ProxyCommand) HandleAuthServer(server *Server) { func (msg *PassCommand) HandleRegServer(server *Server) {
msg.Client().hostname = msg.hostname
}
func (msg *CapCommand) HandleAuthServer(server *Server) {
// TODO
}
func (msg *PassCommand) HandleAuthServer(server *Server) {
client := msg.Client() client := msg.Client()
if msg.err != nil { if msg.err != nil {
client.ErrPasswdMismatch() client.ErrPasswdMismatch()
@ -287,27 +259,70 @@ func (msg *PassCommand) HandleAuthServer(server *Server) {
return return
} }
client.phase = Registration client.authorized = true
} }
func (msg *QuitCommand) HandleAuthServer(server *Server) {
msg.Client().Quit(msg.message)
}
//
// registration commands
//
func (msg *ProxyCommand) HandleRegServer(server *Server) { func (msg *ProxyCommand) HandleRegServer(server *Server) {
msg.Client().hostname = msg.hostname msg.Client().hostname = msg.hostname
} }
func (msg *CapCommand) HandleRegServer(server *Server) { func (msg *CapCommand) HandleRegServer(server *Server) {
// TODO client := msg.Client()
switch msg.subCommand {
case CAP_LS:
client.capState = CapNegotiating
client.Reply("CAP LS * :%s", SupportedCapabilities)
case CAP_LIST:
client.Reply("CAP LIST * :%s", client.capabilities)
case CAP_REQ:
client.capState = CapNegotiating
for capability := range msg.capabilities {
if !SupportedCapabilities[capability] {
client.Reply("CAP NAK * :%s", msg.capabilities)
return
}
}
for capability := range msg.capabilities {
client.capabilities[capability] = true
}
client.Reply("CAP ACK * :%s", msg.capabilities)
case CAP_CLEAR:
format := strings.TrimRight(
strings.Repeat("%s%s ", len(client.capabilities)), " ")
args := make([]interface{}, len(client.capabilities))
index := 0
for capability := range client.capabilities {
args[index] = Disable
args[index+1] = capability
index += 2
delete(client.capabilities, capability)
}
client.Reply("CAP ACK * :"+format, args...)
case CAP_END:
client.capState = CapNegotiated
server.tryRegister(client)
default:
client.ErrInvalidCapCmd(msg.subCommand)
}
} }
func (m *NickCommand) HandleRegServer(s *Server) { func (m *NickCommand) HandleRegServer(s *Server) {
client := m.Client() client := m.Client()
if !client.authorized {
client.ErrPasswdMismatch()
client.Quit("bad password")
return
}
if client.capState == CapNegotiating {
client.capState = CapNegotiated
}
if m.nickname == "" { if m.nickname == "" {
client.ErrNoNicknameGiven() client.ErrNoNicknameGiven()
@ -329,11 +344,22 @@ func (m *NickCommand) HandleRegServer(s *Server) {
} }
func (msg *RFC1459UserCommand) HandleRegServer(server *Server) { func (msg *RFC1459UserCommand) HandleRegServer(server *Server) {
client := msg.Client()
if !client.authorized {
client.ErrPasswdMismatch()
client.Quit("bad password")
return
}
msg.setUserInfo(server) msg.setUserInfo(server)
} }
func (msg *RFC2812UserCommand) HandleRegServer(server *Server) { func (msg *RFC2812UserCommand) HandleRegServer(server *Server) {
client := msg.Client() client := msg.Client()
if !client.authorized {
client.ErrPasswdMismatch()
client.Quit("bad password")
return
}
flags := msg.Flags() flags := msg.Flags()
if len(flags) > 0 { if len(flags) > 0 {
for _, mode := range msg.Flags() { for _, mode := range msg.Flags() {
@ -346,9 +372,14 @@ func (msg *RFC2812UserCommand) HandleRegServer(server *Server) {
func (msg *UserCommand) setUserInfo(server *Server) { func (msg *UserCommand) setUserInfo(server *Server) {
client := msg.Client() client := msg.Client()
if client.capState == CapNegotiating {
client.capState = CapNegotiated
}
server.clients.Remove(client) server.clients.Remove(client)
client.username, client.realname = msg.username, msg.realname client.username, client.realname = msg.username, msg.realname
server.clients.Add(client) server.clients.Add(client)
server.tryRegister(client) server.tryRegister(client)
} }

View File

@ -11,6 +11,30 @@ import (
type UserMaskSet map[string]bool type UserMaskSet map[string]bool
type CapSubCommand string
type Capability string
type CapModifier rune
func (mod CapModifier) String() string {
return string(mod)
}
type CapState uint
type CapabilitySet map[Capability]bool
func (set CapabilitySet) String() string {
strs := make([]string, len(set))
index := 0
for capability := range set {
strs[index] = string(capability)
index += 1
}
return strings.Join(strs, " ")
}
// add, remove, list modes // add, remove, list modes
type ModeOp rune type ModeOp rune
@ -22,7 +46,7 @@ func (op ModeOp) String() string {
type UserMode rune type UserMode rune
func (mode UserMode) String() string { func (mode UserMode) String() string {
return fmt.Sprintf("%c", mode) return string(mode)
} }
type Phase uint type Phase uint
@ -47,7 +71,7 @@ func (code NumericCode) String() string {
type ChannelMode rune type ChannelMode rune
func (mode ChannelMode) String() string { func (mode ChannelMode) String() string {
return fmt.Sprintf("%c", mode) return string(mode)
} }
type ChannelNameMap map[string]*Channel type ChannelNameMap map[string]*Channel