Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. // Copyright (c) 2017 Shivaram Lingamneni <slingamn@cs.stanford.edu>
  2. // released under the MIT license
  3. package irc
  4. import (
  5. "errors"
  6. "sync"
  7. )
  8. var (
  9. InvalidChannelName = errors.New("Invalid channel name")
  10. NoSuchChannel = errors.New("No such channel")
  11. ChannelNameInUse = errors.New("Channel name in use")
  12. )
  13. type channelManagerEntry struct {
  14. channel *Channel
  15. // this is a refcount for joins, so we can avoid a race where we incorrectly
  16. // think the channel is empty (without holding a lock across the entire Channel.Join()
  17. // call)
  18. pendingJoins int
  19. }
  20. // ChannelManager keeps track of all the channels on the server,
  21. // providing synchronization for creation of new channels on first join,
  22. // cleanup of empty channels on last part, and renames.
  23. type ChannelManager struct {
  24. sync.RWMutex // tier 2
  25. chans map[string]*channelManagerEntry
  26. }
  27. // NewChannelManager returns a new ChannelManager.
  28. func NewChannelManager() *ChannelManager {
  29. return &ChannelManager{
  30. chans: make(map[string]*channelManagerEntry),
  31. }
  32. }
  33. // Get returns an existing channel with name equivalent to `name`, or nil
  34. func (cm *ChannelManager) Get(name string) *Channel {
  35. name, err := CasefoldChannel(name)
  36. if err == nil {
  37. cm.RLock()
  38. defer cm.RUnlock()
  39. entry := cm.chans[name]
  40. if entry != nil {
  41. return entry.channel
  42. }
  43. }
  44. return nil
  45. }
  46. // Join causes `client` to join the channel named `name`, creating it if necessary.
  47. func (cm *ChannelManager) Join(client *Client, name string, key string) error {
  48. server := client.server
  49. casefoldedName, err := CasefoldChannel(name)
  50. if err != nil || len(casefoldedName) > server.Limits().ChannelLen {
  51. return NoSuchChannel
  52. }
  53. cm.Lock()
  54. entry := cm.chans[casefoldedName]
  55. if entry == nil {
  56. // XXX give up the lock to check for a registration, then check again
  57. // to see if we need to create the channel. we could solve this by doing LoadChannel
  58. // outside the lock initially on every join, so this is best thought of as an
  59. // optimization to avoid that.
  60. cm.Unlock()
  61. info := client.server.channelRegistry.LoadChannel(casefoldedName)
  62. cm.Lock()
  63. entry = cm.chans[casefoldedName]
  64. if entry == nil {
  65. entry = &channelManagerEntry{
  66. channel: NewChannel(server, name, true, info),
  67. pendingJoins: 0,
  68. }
  69. cm.chans[casefoldedName] = entry
  70. }
  71. }
  72. entry.pendingJoins += 1
  73. cm.Unlock()
  74. entry.channel.Join(client, key)
  75. cm.maybeCleanup(entry, true)
  76. return nil
  77. }
  78. func (cm *ChannelManager) maybeCleanup(entry *channelManagerEntry, afterJoin bool) {
  79. cm.Lock()
  80. defer cm.Unlock()
  81. if entry.channel == nil {
  82. return
  83. }
  84. if afterJoin {
  85. entry.pendingJoins -= 1
  86. }
  87. // TODO(slingamn) right now, registered channels cannot be cleaned up.
  88. // this is because once ChannelManager becomes the source of truth about a channel,
  89. // we can't move the source of truth back to the database unless we do an ACID
  90. // store while holding the ChannelManager's Lock(). This is pending more decisions
  91. // about where the database transaction lock fits into the overall lock model.
  92. if !entry.channel.IsRegistered() && entry.channel.IsEmpty() && entry.pendingJoins == 0 {
  93. // reread the name, handling the case where the channel was renamed
  94. casefoldedName := entry.channel.NameCasefolded()
  95. delete(cm.chans, casefoldedName)
  96. // invalidate the entry (otherwise, a subsequent cleanup attempt could delete
  97. // a valid, distinct entry under casefoldedName):
  98. entry.channel = nil
  99. }
  100. }
  101. // Part parts `client` from the channel named `name`, deleting it if it's empty.
  102. func (cm *ChannelManager) Part(client *Client, name string, message string) error {
  103. casefoldedName, err := CasefoldChannel(name)
  104. if err != nil {
  105. return NoSuchChannel
  106. }
  107. cm.RLock()
  108. entry := cm.chans[casefoldedName]
  109. cm.RUnlock()
  110. if entry == nil {
  111. return NoSuchChannel
  112. }
  113. entry.channel.Part(client, message)
  114. cm.maybeCleanup(entry, false)
  115. return nil
  116. }
  117. // Rename renames a channel (but does not notify the members)
  118. func (cm *ChannelManager) Rename(name string, newname string) error {
  119. cfname, err := CasefoldChannel(name)
  120. if err != nil {
  121. return NoSuchChannel
  122. }
  123. cfnewname, err := CasefoldChannel(newname)
  124. if err != nil {
  125. return InvalidChannelName
  126. }
  127. cm.Lock()
  128. defer cm.Unlock()
  129. if cm.chans[cfnewname] != nil {
  130. return ChannelNameInUse
  131. }
  132. entry := cm.chans[cfname]
  133. if entry == nil {
  134. return NoSuchChannel
  135. }
  136. delete(cm.chans, cfname)
  137. cm.chans[cfnewname] = entry
  138. entry.channel.setName(newname)
  139. entry.channel.setNameCasefolded(cfnewname)
  140. return nil
  141. }
  142. // Len returns the number of channels
  143. func (cm *ChannelManager) Len() int {
  144. cm.RLock()
  145. defer cm.RUnlock()
  146. return len(cm.chans)
  147. }
  148. // Channels returns a slice containing all current channels
  149. func (cm *ChannelManager) Channels() (result []*Channel) {
  150. cm.RLock()
  151. defer cm.RUnlock()
  152. for _, entry := range cm.chans {
  153. result = append(result, entry.channel)
  154. }
  155. return
  156. }