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 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477
  1. // Copyright (c) 2017 Shivaram Lingamneni <slingamn@cs.stanford.edu>
  2. // released under the MIT license
  3. package irc
  4. import (
  5. "sort"
  6. "sync"
  7. "github.com/ergochat/ergo/irc/utils"
  8. )
  9. type channelManagerEntry struct {
  10. channel *Channel
  11. // this is a refcount for joins, so we can avoid a race where we incorrectly
  12. // think the channel is empty (without holding a lock across the entire Channel.Join()
  13. // call)
  14. pendingJoins int
  15. skeleton string
  16. }
  17. // ChannelManager keeps track of all the channels on the server,
  18. // providing synchronization for creation of new channels on first join,
  19. // cleanup of empty channels on last part, and renames.
  20. type ChannelManager struct {
  21. sync.RWMutex // tier 2
  22. // chans is the main data structure, mapping casefolded name -> *Channel
  23. chans map[string]*channelManagerEntry
  24. chansSkeletons utils.StringSet // skeletons of *unregistered* chans
  25. registeredChannels utils.StringSet // casefolds of registered chans
  26. registeredSkeletons utils.StringSet // skeletons of registered chans
  27. purgedChannels utils.StringSet // casefolds of purged chans
  28. server *Server
  29. }
  30. // NewChannelManager returns a new ChannelManager.
  31. func (cm *ChannelManager) Initialize(server *Server) {
  32. cm.chans = make(map[string]*channelManagerEntry)
  33. cm.chansSkeletons = make(utils.StringSet)
  34. cm.server = server
  35. // purging should work even if registration is disabled
  36. cm.purgedChannels = cm.server.channelRegistry.PurgedChannels()
  37. cm.loadRegisteredChannels(server.Config())
  38. }
  39. func (cm *ChannelManager) loadRegisteredChannels(config *Config) {
  40. if !config.Channels.Registration.Enabled {
  41. return
  42. }
  43. var newChannels []*Channel
  44. var collisions []string
  45. defer func() {
  46. for _, ch := range newChannels {
  47. ch.EnsureLoaded()
  48. cm.server.logger.Debug("channels", "initialized registered channel", ch.Name())
  49. }
  50. for _, collision := range collisions {
  51. cm.server.logger.Warning("channels", "registered channel collides with existing channel", collision)
  52. }
  53. }()
  54. rawNames := cm.server.channelRegistry.AllChannels()
  55. cm.Lock()
  56. defer cm.Unlock()
  57. cm.registeredChannels = make(utils.StringSet, len(rawNames))
  58. cm.registeredSkeletons = make(utils.StringSet, len(rawNames))
  59. for _, name := range rawNames {
  60. cfname, err := CasefoldChannel(name)
  61. if err == nil {
  62. cm.registeredChannels.Add(cfname)
  63. }
  64. skeleton, err := Skeleton(name)
  65. if err == nil {
  66. cm.registeredSkeletons.Add(skeleton)
  67. }
  68. if !cm.purgedChannels.Has(cfname) {
  69. if _, ok := cm.chans[cfname]; !ok {
  70. ch := NewChannel(cm.server, name, cfname, true)
  71. cm.chans[cfname] = &channelManagerEntry{
  72. channel: ch,
  73. pendingJoins: 0,
  74. }
  75. newChannels = append(newChannels, ch)
  76. } else {
  77. collisions = append(collisions, name)
  78. }
  79. }
  80. }
  81. }
  82. // Get returns an existing channel with name equivalent to `name`, or nil
  83. func (cm *ChannelManager) Get(name string) (channel *Channel) {
  84. name, err := CasefoldChannel(name)
  85. if err == nil {
  86. cm.RLock()
  87. defer cm.RUnlock()
  88. entry := cm.chans[name]
  89. // if the channel is still loading, pretend we don't have it
  90. if entry != nil && entry.channel.IsLoaded() {
  91. return entry.channel
  92. }
  93. }
  94. return nil
  95. }
  96. // Join causes `client` to join the channel named `name`, creating it if necessary.
  97. func (cm *ChannelManager) Join(client *Client, name string, key string, isSajoin bool, rb *ResponseBuffer) (err error, forward string) {
  98. server := client.server
  99. casefoldedName, err := CasefoldChannel(name)
  100. skeleton, skerr := Skeleton(name)
  101. if err != nil || skerr != nil || len(casefoldedName) > server.Config().Limits.ChannelLen {
  102. return errNoSuchChannel, ""
  103. }
  104. channel, err := func() (*Channel, error) {
  105. cm.Lock()
  106. defer cm.Unlock()
  107. if cm.purgedChannels.Has(casefoldedName) {
  108. return nil, errChannelPurged
  109. }
  110. entry := cm.chans[casefoldedName]
  111. if entry == nil {
  112. registered := cm.registeredChannels.Has(casefoldedName)
  113. // enforce OpOnlyCreation
  114. if !registered && server.Config().Channels.OpOnlyCreation && !client.HasRoleCapabs("chanreg") {
  115. return nil, errInsufficientPrivs
  116. }
  117. // enforce confusables
  118. if !registered && (cm.chansSkeletons.Has(skeleton) || cm.registeredSkeletons.Has(skeleton)) {
  119. return nil, errConfusableIdentifier
  120. }
  121. entry = &channelManagerEntry{
  122. channel: NewChannel(server, name, casefoldedName, registered),
  123. pendingJoins: 0,
  124. }
  125. if !registered {
  126. // for an unregistered channel, we already have the correct unfolded name
  127. // and therefore the final skeleton. for a registered channel, we don't have
  128. // the unfolded name yet (it needs to be loaded from the db), but we already
  129. // have the final skeleton in `registeredSkeletons` so we don't need to track it
  130. cm.chansSkeletons.Add(skeleton)
  131. entry.skeleton = skeleton
  132. }
  133. cm.chans[casefoldedName] = entry
  134. }
  135. entry.pendingJoins += 1
  136. return entry.channel, nil
  137. }()
  138. if err != nil {
  139. return err, ""
  140. }
  141. channel.EnsureLoaded()
  142. err, forward = channel.Join(client, key, isSajoin, rb)
  143. cm.maybeCleanup(channel, true)
  144. return
  145. }
  146. func (cm *ChannelManager) maybeCleanup(channel *Channel, afterJoin bool) {
  147. cm.Lock()
  148. defer cm.Unlock()
  149. cfname := channel.NameCasefolded()
  150. entry := cm.chans[cfname]
  151. if entry == nil || entry.channel != channel {
  152. return
  153. }
  154. cm.maybeCleanupInternal(cfname, entry, afterJoin)
  155. }
  156. func (cm *ChannelManager) maybeCleanupInternal(cfname string, entry *channelManagerEntry, afterJoin bool) {
  157. if afterJoin {
  158. entry.pendingJoins -= 1
  159. }
  160. if entry.pendingJoins == 0 && entry.channel.IsClean() {
  161. delete(cm.chans, cfname)
  162. if entry.skeleton != "" {
  163. delete(cm.chansSkeletons, entry.skeleton)
  164. }
  165. }
  166. }
  167. // Part parts `client` from the channel named `name`, deleting it if it's empty.
  168. func (cm *ChannelManager) Part(client *Client, name string, message string, rb *ResponseBuffer) error {
  169. var channel *Channel
  170. casefoldedName, err := CasefoldChannel(name)
  171. if err != nil {
  172. return errNoSuchChannel
  173. }
  174. cm.RLock()
  175. entry := cm.chans[casefoldedName]
  176. if entry != nil {
  177. channel = entry.channel
  178. }
  179. cm.RUnlock()
  180. if channel == nil {
  181. return errNoSuchChannel
  182. }
  183. channel.Part(client, message, rb)
  184. return nil
  185. }
  186. func (cm *ChannelManager) Cleanup(channel *Channel) {
  187. cm.maybeCleanup(channel, false)
  188. }
  189. func (cm *ChannelManager) SetRegistered(channelName string, account string) (err error) {
  190. if cm.server.Defcon() <= 4 {
  191. return errFeatureDisabled
  192. }
  193. var channel *Channel
  194. cfname, err := CasefoldChannel(channelName)
  195. if err != nil {
  196. return err
  197. }
  198. var entry *channelManagerEntry
  199. defer func() {
  200. if err == nil && channel != nil {
  201. // registration was successful: make the database reflect it
  202. err = channel.Store(IncludeAllAttrs)
  203. }
  204. }()
  205. cm.Lock()
  206. defer cm.Unlock()
  207. entry = cm.chans[cfname]
  208. if entry == nil {
  209. return errNoSuchChannel
  210. }
  211. channel = entry.channel
  212. err = channel.SetRegistered(account)
  213. if err != nil {
  214. return err
  215. }
  216. // transfer the skeleton from chansSkeletons to registeredSkeletons
  217. skeleton := entry.skeleton
  218. delete(cm.chansSkeletons, skeleton)
  219. entry.skeleton = ""
  220. cm.chans[cfname] = entry
  221. cm.registeredChannels.Add(cfname)
  222. cm.registeredSkeletons.Add(skeleton)
  223. return nil
  224. }
  225. func (cm *ChannelManager) SetUnregistered(channelName string, account string) (err error) {
  226. cfname, err := CasefoldChannel(channelName)
  227. if err != nil {
  228. return err
  229. }
  230. info, err := cm.server.channelRegistry.LoadChannel(cfname)
  231. if err != nil {
  232. return err
  233. }
  234. if info.Founder != account {
  235. return errChannelNotOwnedByAccount
  236. }
  237. defer func() {
  238. if err == nil {
  239. err = cm.server.channelRegistry.Delete(info)
  240. }
  241. }()
  242. cm.Lock()
  243. defer cm.Unlock()
  244. entry := cm.chans[cfname]
  245. if entry != nil {
  246. entry.channel.SetUnregistered(account)
  247. delete(cm.registeredChannels, cfname)
  248. // transfer the skeleton from registeredSkeletons to chansSkeletons
  249. if skel, err := Skeleton(entry.channel.Name()); err == nil {
  250. delete(cm.registeredSkeletons, skel)
  251. cm.chansSkeletons.Add(skel)
  252. entry.skeleton = skel
  253. cm.chans[cfname] = entry
  254. }
  255. // #1619: if the channel has 0 members and was only being retained
  256. // because it was registered, clean it up:
  257. cm.maybeCleanupInternal(cfname, entry, false)
  258. }
  259. return nil
  260. }
  261. // Rename renames a channel (but does not notify the members)
  262. func (cm *ChannelManager) Rename(name string, newName string) (err error) {
  263. oldCfname, err := CasefoldChannel(name)
  264. if err != nil {
  265. return errNoSuchChannel
  266. }
  267. newCfname, err := CasefoldChannel(newName)
  268. if err != nil {
  269. return errInvalidChannelName
  270. }
  271. newSkeleton, err := Skeleton(newName)
  272. if err != nil {
  273. return errInvalidChannelName
  274. }
  275. var channel *Channel
  276. var info RegisteredChannel
  277. defer func() {
  278. if channel != nil && info.Founder != "" {
  279. channel.Store(IncludeAllAttrs)
  280. // we just flushed the channel under its new name, therefore this delete
  281. // cannot be overwritten by a write to the old name:
  282. cm.server.channelRegistry.Delete(info)
  283. }
  284. }()
  285. cm.Lock()
  286. defer cm.Unlock()
  287. entry := cm.chans[oldCfname]
  288. if entry == nil || !entry.channel.IsLoaded() {
  289. return errNoSuchChannel
  290. }
  291. channel = entry.channel
  292. info = channel.ExportRegistration(IncludeInitial)
  293. registered := info.Founder != ""
  294. oldSkeleton, err := Skeleton(info.Name)
  295. if err != nil {
  296. return errNoSuchChannel // ugh
  297. }
  298. if newCfname != oldCfname {
  299. if cm.chans[newCfname] != nil || cm.registeredChannels.Has(newCfname) {
  300. return errChannelNameInUse
  301. }
  302. }
  303. if oldSkeleton != newSkeleton {
  304. if cm.chansSkeletons.Has(newSkeleton) || cm.registeredSkeletons.Has(newSkeleton) {
  305. return errConfusableIdentifier
  306. }
  307. }
  308. delete(cm.chans, oldCfname)
  309. if !registered {
  310. entry.skeleton = newSkeleton
  311. }
  312. cm.chans[newCfname] = entry
  313. if registered {
  314. delete(cm.registeredChannels, oldCfname)
  315. cm.registeredChannels.Add(newCfname)
  316. delete(cm.registeredSkeletons, oldSkeleton)
  317. cm.registeredSkeletons.Add(newSkeleton)
  318. } else {
  319. delete(cm.chansSkeletons, oldSkeleton)
  320. cm.chansSkeletons.Add(newSkeleton)
  321. }
  322. entry.channel.Rename(newName, newCfname)
  323. return nil
  324. }
  325. // Len returns the number of channels
  326. func (cm *ChannelManager) Len() int {
  327. cm.RLock()
  328. defer cm.RUnlock()
  329. return len(cm.chans)
  330. }
  331. // Channels returns a slice containing all current channels
  332. func (cm *ChannelManager) Channels() (result []*Channel) {
  333. cm.RLock()
  334. defer cm.RUnlock()
  335. result = make([]*Channel, 0, len(cm.chans))
  336. for _, entry := range cm.chans {
  337. if entry.channel.IsLoaded() {
  338. result = append(result, entry.channel)
  339. }
  340. }
  341. return
  342. }
  343. // Purge marks a channel as purged.
  344. func (cm *ChannelManager) Purge(chname string, record ChannelPurgeRecord) (err error) {
  345. chname, err = CasefoldChannel(chname)
  346. if err != nil {
  347. return errInvalidChannelName
  348. }
  349. skel, err := Skeleton(chname)
  350. if err != nil {
  351. return errInvalidChannelName
  352. }
  353. cm.Lock()
  354. cm.purgedChannels.Add(chname)
  355. entry := cm.chans[chname]
  356. if entry != nil {
  357. delete(cm.chans, chname)
  358. if entry.channel.Founder() != "" {
  359. delete(cm.registeredSkeletons, skel)
  360. } else {
  361. delete(cm.chansSkeletons, skel)
  362. }
  363. }
  364. cm.Unlock()
  365. cm.server.channelRegistry.PurgeChannel(chname, record)
  366. if entry != nil {
  367. entry.channel.Purge("")
  368. }
  369. return nil
  370. }
  371. // IsPurged queries whether a channel is purged.
  372. func (cm *ChannelManager) IsPurged(chname string) (result bool) {
  373. chname, err := CasefoldChannel(chname)
  374. if err != nil {
  375. return false
  376. }
  377. cm.RLock()
  378. result = cm.purgedChannels.Has(chname)
  379. cm.RUnlock()
  380. return
  381. }
  382. // Unpurge deletes a channel's purged status.
  383. func (cm *ChannelManager) Unpurge(chname string) (err error) {
  384. chname, err = CasefoldChannel(chname)
  385. if err != nil {
  386. return errNoSuchChannel
  387. }
  388. cm.Lock()
  389. found := cm.purgedChannels.Has(chname)
  390. delete(cm.purgedChannels, chname)
  391. cm.Unlock()
  392. cm.server.channelRegistry.UnpurgeChannel(chname)
  393. if !found {
  394. return errNoSuchChannel
  395. }
  396. return nil
  397. }
  398. func (cm *ChannelManager) ListPurged() (result []string) {
  399. cm.RLock()
  400. result = make([]string, 0, len(cm.purgedChannels))
  401. for c := range cm.purgedChannels {
  402. result = append(result, c)
  403. }
  404. cm.RUnlock()
  405. sort.Strings(result)
  406. return
  407. }
  408. func (cm *ChannelManager) UnfoldName(cfname string) (result string) {
  409. cm.RLock()
  410. entry := cm.chans[cfname]
  411. cm.RUnlock()
  412. if entry != nil && entry.channel.IsLoaded() {
  413. return entry.channel.Name()
  414. }
  415. return cfname
  416. }