mirror of
https://github.com/ergochat/ergo.git
synced 2025-01-08 19:22:53 +01:00
Fix various locks around joining, kicking and quitting
This commit is contained in:
parent
6f7c683247
commit
c39bebc696
@ -68,11 +68,23 @@ func NewChannel(s *Server, name string, addDefaultModes bool) *Channel {
|
|||||||
func (channel *Channel) IsEmpty() bool {
|
func (channel *Channel) IsEmpty() bool {
|
||||||
channel.membersMutex.RLock()
|
channel.membersMutex.RLock()
|
||||||
defer channel.membersMutex.RUnlock()
|
defer channel.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
return channel.isEmptyNoMutex()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (channel *Channel) isEmptyNoMutex() bool {
|
||||||
return len(channel.members) == 0
|
return len(channel.members) == 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func (channel *Channel) Names(client *Client) {
|
func (channel *Channel) Names(client *Client) {
|
||||||
currentNicks := channel.Nicks(client)
|
channel.membersMutex.RLock()
|
||||||
|
defer channel.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
channel.namesNoMutex(client)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (channel *Channel) namesNoMutex(client *Client) {
|
||||||
|
currentNicks := channel.nicksNoMutex(client)
|
||||||
// assemble and send replies
|
// assemble and send replies
|
||||||
maxNamLen := 480 - len(client.server.name) - len(client.nick)
|
maxNamLen := 480 - len(client.server.name) - len(client.nick)
|
||||||
var buffer string
|
var buffer string
|
||||||
@ -101,6 +113,12 @@ func (channel *Channel) ClientIsAtLeast(client *Client, permission ChannelMode)
|
|||||||
channel.membersMutex.RLock()
|
channel.membersMutex.RLock()
|
||||||
defer channel.membersMutex.RUnlock()
|
defer channel.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
return channel.clientIsAtLeastNoMutex(client, permission)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (channel *Channel) clientIsAtLeastNoMutex(client *Client, permission ChannelMode) bool {
|
||||||
|
// requires RLock()
|
||||||
|
|
||||||
// get voice, since it's not a part of ChannelPrivModes
|
// get voice, since it's not a part of ChannelPrivModes
|
||||||
if channel.members.HasMode(client, permission) {
|
if channel.members.HasMode(client, permission) {
|
||||||
return true
|
return true
|
||||||
@ -141,10 +159,7 @@ func (modes ChannelModeSet) Prefixes(isMultiPrefix bool) string {
|
|||||||
return prefixes
|
return prefixes
|
||||||
}
|
}
|
||||||
|
|
||||||
func (channel *Channel) Nicks(target *Client) []string {
|
func (channel *Channel) nicksNoMutex(target *Client) []string {
|
||||||
channel.membersMutex.RLock()
|
|
||||||
defer channel.membersMutex.RUnlock()
|
|
||||||
|
|
||||||
isMultiPrefix := (target != nil) && target.capabilities[MultiPrefix]
|
isMultiPrefix := (target != nil) && target.capabilities[MultiPrefix]
|
||||||
isUserhostInNames := (target != nil) && target.capabilities[UserhostInNames]
|
isUserhostInNames := (target != nil) && target.capabilities[UserhostInNames]
|
||||||
nicks := make([]string, len(channel.members))
|
nicks := make([]string, len(channel.members))
|
||||||
@ -218,12 +233,11 @@ func (channel *Channel) CheckKey(key string) bool {
|
|||||||
|
|
||||||
func (channel *Channel) Join(client *Client, key string) {
|
func (channel *Channel) Join(client *Client, key string) {
|
||||||
channel.membersMutex.Lock()
|
channel.membersMutex.Lock()
|
||||||
defer channel.membersMutex.Unlock()
|
|
||||||
|
|
||||||
if channel.members.Has(client) {
|
if channel.members.Has(client) {
|
||||||
// already joined, no message?
|
// already joined, no message?
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
channel.membersMutex.Unlock()
|
||||||
|
|
||||||
if channel.IsFull() {
|
if channel.IsFull() {
|
||||||
client.Send(nil, client.server.name, ERR_CHANNELISFULL, channel.name, "Cannot join channel (+l)")
|
client.Send(nil, client.server.name, ERR_CHANNELISFULL, channel.name, "Cannot join channel (+l)")
|
||||||
@ -241,6 +255,8 @@ func (channel *Channel) Join(client *Client, key string) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
channel.membersMutex.Lock()
|
||||||
|
defer channel.membersMutex.Unlock()
|
||||||
if channel.lists[BanMask].Match(client.nickMaskCasefolded) &&
|
if channel.lists[BanMask].Match(client.nickMaskCasefolded) &&
|
||||||
!isInvited &&
|
!isInvited &&
|
||||||
!channel.lists[ExceptMask].Match(client.nickMaskCasefolded) {
|
!channel.lists[ExceptMask].Match(client.nickMaskCasefolded) {
|
||||||
@ -270,8 +286,8 @@ func (channel *Channel) Join(client *Client, key string) {
|
|||||||
} else {
|
} else {
|
||||||
client.Send(nil, client.nickMaskString, "JOIN", channel.name)
|
client.Send(nil, client.nickMaskString, "JOIN", channel.name)
|
||||||
}
|
}
|
||||||
channel.GetTopic(client)
|
channel.getTopicNoMutex(client) // we already have Lock
|
||||||
channel.Names(client)
|
channel.namesNoMutex(client)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (channel *Channel) Part(client *Client, message string) {
|
func (channel *Channel) Part(client *Client, message string) {
|
||||||
@ -293,6 +309,10 @@ func (channel *Channel) GetTopic(client *Client) {
|
|||||||
channel.membersMutex.RLock()
|
channel.membersMutex.RLock()
|
||||||
defer channel.membersMutex.RUnlock()
|
defer channel.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
channel.getTopicNoMutex(client)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (channel *Channel) getTopicNoMutex(client *Client) {
|
||||||
if !channel.members.Has(client) {
|
if !channel.members.Has(client) {
|
||||||
client.Send(nil, client.server.name, ERR_NOTONCHANNEL, client.nick, channel.name, "You're not on that channel")
|
client.Send(nil, client.server.name, ERR_NOTONCHANNEL, client.nick, channel.name, "You're not on that channel")
|
||||||
return
|
return
|
||||||
@ -507,23 +527,26 @@ func (channel *Channel) Quit(client *Client) {
|
|||||||
channel.membersMutex.Lock()
|
channel.membersMutex.Lock()
|
||||||
defer channel.membersMutex.Unlock()
|
defer channel.membersMutex.Unlock()
|
||||||
|
|
||||||
|
channel.quitNoMutex(client)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (channel *Channel) quitNoMutex(client *Client) {
|
||||||
channel.members.Remove(client)
|
channel.members.Remove(client)
|
||||||
client.channels.Remove(channel)
|
client.channels.Remove(channel)
|
||||||
|
|
||||||
if channel.IsEmpty() {
|
if channel.isEmptyNoMutex() {
|
||||||
channel.server.channels.Remove(channel)
|
channel.server.channels.Remove(channel)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (channel *Channel) Kick(client *Client, target *Client, comment string) {
|
func (channel *Channel) kickNoMutex(client *Client, target *Client, comment string) {
|
||||||
channel.membersMutex.Lock()
|
// needs a Lock()
|
||||||
defer channel.membersMutex.Unlock()
|
|
||||||
|
|
||||||
if !(client.flags[Operator] || channel.members.Has(client)) {
|
if !(client.flags[Operator] || channel.members.Has(client)) {
|
||||||
client.Send(nil, client.server.name, ERR_NOTONCHANNEL, channel.name, "You're not on that channel")
|
client.Send(nil, client.server.name, ERR_NOTONCHANNEL, channel.name, "You're not on that channel")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if !channel.ClientIsAtLeast(client, ChannelOperator) {
|
if !channel.clientIsAtLeastNoMutex(client, ChannelOperator) {
|
||||||
client.Send(nil, client.server.name, ERR_CANNOTSENDTOCHAN, channel.name, "Cannot send to channel")
|
client.Send(nil, client.server.name, ERR_CANNOTSENDTOCHAN, channel.name, "Cannot send to channel")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -539,7 +562,7 @@ func (channel *Channel) Kick(client *Client, target *Client, comment string) {
|
|||||||
for member := range channel.members {
|
for member := range channel.members {
|
||||||
member.Send(nil, client.nickMaskString, "KICK", channel.name, target.nick, comment)
|
member.Send(nil, client.nickMaskString, "KICK", channel.name, target.nick, comment)
|
||||||
}
|
}
|
||||||
channel.Quit(target)
|
channel.quitNoMutex(target)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (channel *Channel) Invite(invitee *Client, inviter *Client) {
|
func (channel *Channel) Invite(invitee *Client, inviter *Client) {
|
||||||
|
@ -1381,8 +1381,7 @@ func kickHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
|
|||||||
// make sure client has privs to kick the given user
|
// make sure client has privs to kick the given user
|
||||||
//TODO(dan): split this into a separate function that checks if users have privs
|
//TODO(dan): split this into a separate function that checks if users have privs
|
||||||
// over other users, useful for things like -aoh as well
|
// over other users, useful for things like -aoh as well
|
||||||
channel.membersMutex.RLock()
|
channel.membersMutex.Lock()
|
||||||
defer channel.membersMutex.RUnlock()
|
|
||||||
|
|
||||||
var hasPrivs bool
|
var hasPrivs bool
|
||||||
for _, mode := range ChannelPrivModes {
|
for _, mode := range ChannelPrivModes {
|
||||||
@ -1404,10 +1403,12 @@ func kickHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
|
|||||||
if comment == "" {
|
if comment == "" {
|
||||||
comment = nickname
|
comment = nickname
|
||||||
}
|
}
|
||||||
channel.Kick(client, target, comment)
|
channel.kickNoMutex(client, target, comment)
|
||||||
} else {
|
} else {
|
||||||
client.Send(nil, client.server.name, ERR_CHANOPRIVSNEEDED, chname, "You're not a channel operator")
|
client.Send(nil, client.server.name, ERR_CHANOPRIVSNEEDED, chname, "You're not a channel operator")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
channel.membersMutex.Unlock()
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user