You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

channelmanager.go 6.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. // Copyright (c) 2017 Shivaram Lingamneni <slingamn@cs.stanford.edu>
  2. // released under the MIT license
  3. package irc
  4. import (
  5. "sync"
  6. )
  7. type channelManagerEntry struct {
  8. channel *Channel
  9. // this is a refcount for joins, so we can avoid a race where we incorrectly
  10. // think the channel is empty (without holding a lock across the entire Channel.Join()
  11. // call)
  12. pendingJoins int
  13. }
  14. // ChannelManager keeps track of all the channels on the server,
  15. // providing synchronization for creation of new channels on first join,
  16. // cleanup of empty channels on last part, and renames.
  17. type ChannelManager struct {
  18. sync.RWMutex // tier 2
  19. chans map[string]*channelManagerEntry
  20. registeredChannels map[string]bool
  21. server *Server
  22. }
  23. // NewChannelManager returns a new ChannelManager.
  24. func (cm *ChannelManager) Initialize(server *Server) {
  25. cm.chans = make(map[string]*channelManagerEntry)
  26. cm.server = server
  27. if server.Config().Channels.Registration.Enabled {
  28. cm.loadRegisteredChannels()
  29. }
  30. }
  31. func (cm *ChannelManager) loadRegisteredChannels() {
  32. registeredChannels := cm.server.channelRegistry.AllChannels()
  33. cm.Lock()
  34. defer cm.Unlock()
  35. cm.registeredChannels = registeredChannels
  36. }
  37. // Get returns an existing channel with name equivalent to `name`, or nil
  38. func (cm *ChannelManager) Get(name string) (channel *Channel) {
  39. name, err := CasefoldChannel(name)
  40. if err == nil {
  41. cm.RLock()
  42. defer cm.RUnlock()
  43. entry := cm.chans[name]
  44. // if the channel is still loading, pretend we don't have it
  45. if entry != nil && entry.channel.IsLoaded() {
  46. return entry.channel
  47. }
  48. }
  49. return nil
  50. }
  51. // Join causes `client` to join the channel named `name`, creating it if necessary.
  52. func (cm *ChannelManager) Join(client *Client, name string, key string, isSajoin bool, rb *ResponseBuffer) error {
  53. server := client.server
  54. casefoldedName, err := CasefoldChannel(name)
  55. if err != nil || len(casefoldedName) > server.Config().Limits.ChannelLen {
  56. return errNoSuchChannel
  57. }
  58. channel := func() *Channel {
  59. cm.Lock()
  60. defer cm.Unlock()
  61. entry := cm.chans[casefoldedName]
  62. if entry == nil {
  63. registered := cm.registeredChannels[casefoldedName]
  64. // enforce OpOnlyCreation
  65. if !registered && server.Config().Channels.OpOnlyCreation && !client.HasRoleCapabs("chanreg") {
  66. return nil
  67. }
  68. entry = &channelManagerEntry{
  69. channel: NewChannel(server, name, registered),
  70. pendingJoins: 0,
  71. }
  72. cm.chans[casefoldedName] = entry
  73. }
  74. entry.pendingJoins += 1
  75. return entry.channel
  76. }()
  77. if channel == nil {
  78. return errNoSuchChannel
  79. }
  80. channel.EnsureLoaded()
  81. channel.Join(client, key, isSajoin, rb)
  82. cm.maybeCleanup(channel, true)
  83. return nil
  84. }
  85. func (cm *ChannelManager) maybeCleanup(channel *Channel, afterJoin bool) {
  86. cm.Lock()
  87. defer cm.Unlock()
  88. nameCasefolded := channel.NameCasefolded()
  89. entry := cm.chans[nameCasefolded]
  90. if entry == nil || entry.channel != channel {
  91. return
  92. }
  93. if afterJoin {
  94. entry.pendingJoins -= 1
  95. }
  96. if entry.pendingJoins == 0 && entry.channel.IsClean() {
  97. delete(cm.chans, nameCasefolded)
  98. }
  99. }
  100. // Part parts `client` from the channel named `name`, deleting it if it's empty.
  101. func (cm *ChannelManager) Part(client *Client, name string, message string, rb *ResponseBuffer) error {
  102. var channel *Channel
  103. casefoldedName, err := CasefoldChannel(name)
  104. if err != nil {
  105. return errNoSuchChannel
  106. }
  107. cm.RLock()
  108. entry := cm.chans[casefoldedName]
  109. if entry != nil {
  110. channel = entry.channel
  111. }
  112. cm.RUnlock()
  113. if channel == nil {
  114. return errNoSuchChannel
  115. }
  116. channel.Part(client, message, rb)
  117. return nil
  118. }
  119. func (cm *ChannelManager) Cleanup(channel *Channel) {
  120. cm.maybeCleanup(channel, false)
  121. }
  122. func (cm *ChannelManager) SetRegistered(channelName string, account string) (err error) {
  123. var channel *Channel
  124. cfname, err := CasefoldChannel(channelName)
  125. if err != nil {
  126. return err
  127. }
  128. var entry *channelManagerEntry
  129. defer func() {
  130. if err == nil && channel != nil {
  131. // registration was successful: make the database reflect it
  132. err = channel.Store(IncludeAllChannelAttrs)
  133. }
  134. }()
  135. cm.Lock()
  136. defer cm.Unlock()
  137. entry = cm.chans[cfname]
  138. if entry == nil {
  139. return errNoSuchChannel
  140. }
  141. channel = entry.channel
  142. err = channel.SetRegistered(account)
  143. if err != nil {
  144. return err
  145. }
  146. cm.registeredChannels[cfname] = true
  147. return nil
  148. }
  149. func (cm *ChannelManager) SetUnregistered(channelName string, account string) (err error) {
  150. cfname, err := CasefoldChannel(channelName)
  151. if err != nil {
  152. return err
  153. }
  154. var info RegisteredChannel
  155. defer func() {
  156. if err == nil {
  157. err = cm.server.channelRegistry.Delete(info)
  158. }
  159. }()
  160. cm.Lock()
  161. defer cm.Unlock()
  162. entry := cm.chans[cfname]
  163. if entry == nil {
  164. return errNoSuchChannel
  165. }
  166. info = entry.channel.ExportRegistration(0)
  167. if info.Founder != account {
  168. return errChannelNotOwnedByAccount
  169. }
  170. entry.channel.SetUnregistered(account)
  171. delete(cm.registeredChannels, cfname)
  172. return nil
  173. }
  174. // Rename renames a channel (but does not notify the members)
  175. func (cm *ChannelManager) Rename(name string, newname string) (err error) {
  176. cfname, err := CasefoldChannel(name)
  177. if err != nil {
  178. return errNoSuchChannel
  179. }
  180. cfnewname, err := CasefoldChannel(newname)
  181. if err != nil {
  182. return errInvalidChannelName
  183. }
  184. var channel *Channel
  185. var info RegisteredChannel
  186. defer func() {
  187. if channel != nil && info.Founder != "" {
  188. channel.Store(IncludeAllChannelAttrs)
  189. // we just flushed the channel under its new name, therefore this delete
  190. // cannot be overwritten by a write to the old name:
  191. cm.server.channelRegistry.Delete(info)
  192. }
  193. }()
  194. cm.Lock()
  195. defer cm.Unlock()
  196. if cm.chans[cfnewname] != nil || cm.registeredChannels[cfnewname] {
  197. return errChannelNameInUse
  198. }
  199. entry := cm.chans[cfname]
  200. if entry == nil {
  201. return errNoSuchChannel
  202. }
  203. channel = entry.channel
  204. info = channel.ExportRegistration(IncludeInitial)
  205. delete(cm.chans, cfname)
  206. cm.chans[cfnewname] = entry
  207. if cm.registeredChannels[cfname] {
  208. delete(cm.registeredChannels, cfname)
  209. cm.registeredChannels[cfnewname] = true
  210. }
  211. entry.channel.Rename(newname, cfnewname)
  212. return nil
  213. }
  214. // Len returns the number of channels
  215. func (cm *ChannelManager) Len() int {
  216. cm.RLock()
  217. defer cm.RUnlock()
  218. return len(cm.chans)
  219. }
  220. // Channels returns a slice containing all current channels
  221. func (cm *ChannelManager) Channels() (result []*Channel) {
  222. cm.RLock()
  223. defer cm.RUnlock()
  224. result = make([]*Channel, 0, len(cm.chans))
  225. for _, entry := range cm.chans {
  226. if entry.channel.IsLoaded() {
  227. result = append(result, entry.channel)
  228. }
  229. }
  230. return
  231. }