Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

channelreg.go 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. // Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
  2. // released under the MIT license
  3. package irc
  4. import (
  5. "encoding/json"
  6. "fmt"
  7. "strconv"
  8. "strings"
  9. "time"
  10. "github.com/tidwall/buntdb"
  11. "github.com/oragono/oragono/irc/modes"
  12. "github.com/oragono/oragono/irc/utils"
  13. )
  14. // this is exclusively the *persistence* layer for channel registration;
  15. // channel creation/tracking/destruction is in channelmanager.go
  16. const (
  17. keyChannelExists = "channel.exists %s"
  18. keyChannelName = "channel.name %s" // stores the 'preferred name' of the channel, not casemapped
  19. keyChannelRegTime = "channel.registered.time %s"
  20. keyChannelFounder = "channel.founder %s"
  21. keyChannelTopic = "channel.topic %s"
  22. keyChannelTopicSetBy = "channel.topic.setby %s"
  23. keyChannelTopicSetTime = "channel.topic.settime %s"
  24. keyChannelBanlist = "channel.banlist %s"
  25. keyChannelExceptlist = "channel.exceptlist %s"
  26. keyChannelInvitelist = "channel.invitelist %s"
  27. keyChannelPassword = "channel.key %s"
  28. keyChannelModes = "channel.modes %s"
  29. keyChannelAccountToUMode = "channel.accounttoumode %s"
  30. keyChannelUserLimit = "channel.userlimit %s"
  31. keyChannelSettings = "channel.settings %s"
  32. keyChannelPurged = "channel.purged %s"
  33. )
  34. var (
  35. channelKeyStrings = []string{
  36. keyChannelExists,
  37. keyChannelName,
  38. keyChannelRegTime,
  39. keyChannelFounder,
  40. keyChannelTopic,
  41. keyChannelTopicSetBy,
  42. keyChannelTopicSetTime,
  43. keyChannelBanlist,
  44. keyChannelExceptlist,
  45. keyChannelInvitelist,
  46. keyChannelPassword,
  47. keyChannelModes,
  48. keyChannelAccountToUMode,
  49. keyChannelUserLimit,
  50. keyChannelSettings,
  51. }
  52. )
  53. // these are bit flags indicating what part of the channel status is "dirty"
  54. // and needs to be read from memory and written to the db
  55. const (
  56. IncludeInitial uint = 1 << iota
  57. IncludeTopic
  58. IncludeModes
  59. IncludeLists
  60. IncludeSettings
  61. )
  62. // this is an OR of all possible flags
  63. const (
  64. IncludeAllAttrs = ^uint(0)
  65. )
  66. // RegisteredChannel holds details about a given registered channel.
  67. type RegisteredChannel struct {
  68. // Name of the channel.
  69. Name string
  70. // Casefolded name of the channel.
  71. NameCasefolded string
  72. // RegisteredAt represents the time that the channel was registered.
  73. RegisteredAt time.Time
  74. // Founder indicates the founder of the channel.
  75. Founder string
  76. // Topic represents the channel topic.
  77. Topic string
  78. // TopicSetBy represents the host that set the topic.
  79. TopicSetBy string
  80. // TopicSetTime represents the time the topic was set.
  81. TopicSetTime time.Time
  82. // Modes represents the channel modes
  83. Modes []modes.Mode
  84. // Key represents the channel key / password
  85. Key string
  86. // UserLimit is the user limit (0 for no limit)
  87. UserLimit int
  88. // AccountToUMode maps user accounts to their persistent channel modes (e.g., +q, +h)
  89. AccountToUMode map[string]modes.Mode
  90. // Bans represents the bans set on the channel.
  91. Bans map[string]MaskInfo
  92. // Excepts represents the exceptions set on the channel.
  93. Excepts map[string]MaskInfo
  94. // Invites represents the invite exceptions set on the channel.
  95. Invites map[string]MaskInfo
  96. // Settings are the chanserv-modifiable settings
  97. Settings ChannelSettings
  98. }
  99. type ChannelPurgeRecord struct {
  100. Oper string
  101. PurgedAt time.Time
  102. Reason string
  103. }
  104. // ChannelRegistry manages registered channels.
  105. type ChannelRegistry struct {
  106. server *Server
  107. }
  108. // NewChannelRegistry returns a new ChannelRegistry.
  109. func (reg *ChannelRegistry) Initialize(server *Server) {
  110. reg.server = server
  111. }
  112. // AllChannels returns the uncasefolded names of all registered channels.
  113. func (reg *ChannelRegistry) AllChannels() (result []string) {
  114. prefix := fmt.Sprintf(keyChannelName, "")
  115. reg.server.store.View(func(tx *buntdb.Tx) error {
  116. return tx.AscendGreaterOrEqual("", prefix, func(key, value string) bool {
  117. if !strings.HasPrefix(key, prefix) {
  118. return false
  119. }
  120. result = append(result, value)
  121. return true
  122. })
  123. })
  124. return
  125. }
  126. // PurgedChannels returns the set of all casefolded channel names that have been purged
  127. func (reg *ChannelRegistry) PurgedChannels() (result utils.StringSet) {
  128. result = make(utils.StringSet)
  129. prefix := fmt.Sprintf(keyChannelPurged, "")
  130. reg.server.store.View(func(tx *buntdb.Tx) error {
  131. return tx.AscendGreaterOrEqual("", prefix, func(key, value string) bool {
  132. if !strings.HasPrefix(key, prefix) {
  133. return false
  134. }
  135. channel := strings.TrimPrefix(key, prefix)
  136. result.Add(channel)
  137. return true
  138. })
  139. })
  140. return
  141. }
  142. // StoreChannel obtains a consistent view of a channel, then persists it to the store.
  143. func (reg *ChannelRegistry) StoreChannel(info RegisteredChannel, includeFlags uint) (err error) {
  144. if !reg.server.ChannelRegistrationEnabled() {
  145. return
  146. }
  147. if info.Founder == "" {
  148. // sanity check, don't try to store an unregistered channel
  149. return
  150. }
  151. reg.server.store.Update(func(tx *buntdb.Tx) error {
  152. reg.saveChannel(tx, info, includeFlags)
  153. return nil
  154. })
  155. return nil
  156. }
  157. // LoadChannel loads a channel from the store.
  158. func (reg *ChannelRegistry) LoadChannel(nameCasefolded string) (info RegisteredChannel, err error) {
  159. if !reg.server.ChannelRegistrationEnabled() {
  160. err = errFeatureDisabled
  161. return
  162. }
  163. channelKey := nameCasefolded
  164. // nice to have: do all JSON (de)serialization outside of the buntdb transaction
  165. err = reg.server.store.View(func(tx *buntdb.Tx) error {
  166. _, dberr := tx.Get(fmt.Sprintf(keyChannelExists, channelKey))
  167. if dberr == buntdb.ErrNotFound {
  168. // chan does not already exist, return
  169. return errNoSuchChannel
  170. }
  171. // channel exists, load it
  172. name, _ := tx.Get(fmt.Sprintf(keyChannelName, channelKey))
  173. regTime, _ := tx.Get(fmt.Sprintf(keyChannelRegTime, channelKey))
  174. regTimeInt, _ := strconv.ParseInt(regTime, 10, 64)
  175. founder, _ := tx.Get(fmt.Sprintf(keyChannelFounder, channelKey))
  176. topic, _ := tx.Get(fmt.Sprintf(keyChannelTopic, channelKey))
  177. topicSetBy, _ := tx.Get(fmt.Sprintf(keyChannelTopicSetBy, channelKey))
  178. var topicSetTime time.Time
  179. topicSetTimeStr, _ := tx.Get(fmt.Sprintf(keyChannelTopicSetTime, channelKey))
  180. if topicSetTimeInt, topicSetTimeErr := strconv.ParseInt(topicSetTimeStr, 10, 64); topicSetTimeErr == nil {
  181. topicSetTime = time.Unix(0, topicSetTimeInt).UTC()
  182. }
  183. password, _ := tx.Get(fmt.Sprintf(keyChannelPassword, channelKey))
  184. modeString, _ := tx.Get(fmt.Sprintf(keyChannelModes, channelKey))
  185. userLimitString, _ := tx.Get(fmt.Sprintf(keyChannelUserLimit, channelKey))
  186. banlistString, _ := tx.Get(fmt.Sprintf(keyChannelBanlist, channelKey))
  187. exceptlistString, _ := tx.Get(fmt.Sprintf(keyChannelExceptlist, channelKey))
  188. invitelistString, _ := tx.Get(fmt.Sprintf(keyChannelInvitelist, channelKey))
  189. accountToUModeString, _ := tx.Get(fmt.Sprintf(keyChannelAccountToUMode, channelKey))
  190. settingsString, _ := tx.Get(fmt.Sprintf(keyChannelSettings, channelKey))
  191. modeSlice := make([]modes.Mode, len(modeString))
  192. for i, mode := range modeString {
  193. modeSlice[i] = modes.Mode(mode)
  194. }
  195. userLimit, _ := strconv.Atoi(userLimitString)
  196. var banlist map[string]MaskInfo
  197. _ = json.Unmarshal([]byte(banlistString), &banlist)
  198. var exceptlist map[string]MaskInfo
  199. _ = json.Unmarshal([]byte(exceptlistString), &exceptlist)
  200. var invitelist map[string]MaskInfo
  201. _ = json.Unmarshal([]byte(invitelistString), &invitelist)
  202. accountToUMode := make(map[string]modes.Mode)
  203. _ = json.Unmarshal([]byte(accountToUModeString), &accountToUMode)
  204. var settings ChannelSettings
  205. _ = json.Unmarshal([]byte(settingsString), &settings)
  206. info = RegisteredChannel{
  207. Name: name,
  208. NameCasefolded: nameCasefolded,
  209. RegisteredAt: time.Unix(0, regTimeInt).UTC(),
  210. Founder: founder,
  211. Topic: topic,
  212. TopicSetBy: topicSetBy,
  213. TopicSetTime: topicSetTime,
  214. Key: password,
  215. Modes: modeSlice,
  216. Bans: banlist,
  217. Excepts: exceptlist,
  218. Invites: invitelist,
  219. AccountToUMode: accountToUMode,
  220. UserLimit: int(userLimit),
  221. Settings: settings,
  222. }
  223. return nil
  224. })
  225. return
  226. }
  227. // Delete deletes a channel corresponding to `info`. If no such channel
  228. // is present in the database, no error is returned.
  229. func (reg *ChannelRegistry) Delete(info RegisteredChannel) (err error) {
  230. if !reg.server.ChannelRegistrationEnabled() {
  231. return
  232. }
  233. reg.server.store.Update(func(tx *buntdb.Tx) error {
  234. reg.deleteChannel(tx, info.NameCasefolded, info)
  235. return nil
  236. })
  237. return nil
  238. }
  239. // delete a channel, unless it was overwritten by another registration of the same channel
  240. func (reg *ChannelRegistry) deleteChannel(tx *buntdb.Tx, key string, info RegisteredChannel) {
  241. _, err := tx.Get(fmt.Sprintf(keyChannelExists, key))
  242. if err == nil {
  243. regTime, _ := tx.Get(fmt.Sprintf(keyChannelRegTime, key))
  244. regTimeInt, _ := strconv.ParseInt(regTime, 10, 64)
  245. registeredAt := time.Unix(0, regTimeInt).UTC()
  246. founder, _ := tx.Get(fmt.Sprintf(keyChannelFounder, key))
  247. // to see if we're deleting the right channel, confirm the founder and the registration time
  248. if founder == info.Founder && registeredAt.Equal(info.RegisteredAt) {
  249. for _, keyFmt := range channelKeyStrings {
  250. tx.Delete(fmt.Sprintf(keyFmt, key))
  251. }
  252. // remove this channel from the client's list of registered channels
  253. channelsKey := fmt.Sprintf(keyAccountChannels, info.Founder)
  254. channelsStr, err := tx.Get(channelsKey)
  255. if err == buntdb.ErrNotFound {
  256. return
  257. }
  258. registeredChannels := unmarshalRegisteredChannels(channelsStr)
  259. var nowRegisteredChannels []string
  260. for _, channel := range registeredChannels {
  261. if channel != key {
  262. nowRegisteredChannels = append(nowRegisteredChannels, channel)
  263. }
  264. }
  265. tx.Set(channelsKey, strings.Join(nowRegisteredChannels, ","), nil)
  266. }
  267. }
  268. }
  269. func (reg *ChannelRegistry) updateAccountToChannelMapping(tx *buntdb.Tx, channelInfo RegisteredChannel) {
  270. channelKey := channelInfo.NameCasefolded
  271. chanFounderKey := fmt.Sprintf(keyChannelFounder, channelKey)
  272. founder, existsErr := tx.Get(chanFounderKey)
  273. if existsErr == buntdb.ErrNotFound || founder != channelInfo.Founder {
  274. // add to new founder's list
  275. accountChannelsKey := fmt.Sprintf(keyAccountChannels, channelInfo.Founder)
  276. alreadyChannels, _ := tx.Get(accountChannelsKey)
  277. newChannels := channelKey // this is the casefolded channel name
  278. if alreadyChannels != "" {
  279. newChannels = fmt.Sprintf("%s,%s", alreadyChannels, newChannels)
  280. }
  281. tx.Set(accountChannelsKey, newChannels, nil)
  282. }
  283. if existsErr == nil && founder != channelInfo.Founder {
  284. // remove from old founder's list
  285. accountChannelsKey := fmt.Sprintf(keyAccountChannels, founder)
  286. alreadyChannelsRaw, _ := tx.Get(accountChannelsKey)
  287. var newChannels []string
  288. if alreadyChannelsRaw != "" {
  289. for _, chname := range strings.Split(alreadyChannelsRaw, ",") {
  290. if chname != channelInfo.NameCasefolded {
  291. newChannels = append(newChannels, chname)
  292. }
  293. }
  294. }
  295. tx.Set(accountChannelsKey, strings.Join(newChannels, ","), nil)
  296. }
  297. }
  298. // saveChannel saves a channel to the store.
  299. func (reg *ChannelRegistry) saveChannel(tx *buntdb.Tx, channelInfo RegisteredChannel, includeFlags uint) {
  300. channelKey := channelInfo.NameCasefolded
  301. // maintain the mapping of account -> registered channels
  302. reg.updateAccountToChannelMapping(tx, channelInfo)
  303. if includeFlags&IncludeInitial != 0 {
  304. tx.Set(fmt.Sprintf(keyChannelExists, channelKey), "1", nil)
  305. tx.Set(fmt.Sprintf(keyChannelName, channelKey), channelInfo.Name, nil)
  306. tx.Set(fmt.Sprintf(keyChannelRegTime, channelKey), strconv.FormatInt(channelInfo.RegisteredAt.UnixNano(), 10), nil)
  307. tx.Set(fmt.Sprintf(keyChannelFounder, channelKey), channelInfo.Founder, nil)
  308. }
  309. if includeFlags&IncludeTopic != 0 {
  310. tx.Set(fmt.Sprintf(keyChannelTopic, channelKey), channelInfo.Topic, nil)
  311. var topicSetTimeStr string
  312. if !channelInfo.TopicSetTime.IsZero() {
  313. topicSetTimeStr = strconv.FormatInt(channelInfo.TopicSetTime.UnixNano(), 10)
  314. }
  315. tx.Set(fmt.Sprintf(keyChannelTopicSetTime, channelKey), topicSetTimeStr, nil)
  316. tx.Set(fmt.Sprintf(keyChannelTopicSetBy, channelKey), channelInfo.TopicSetBy, nil)
  317. }
  318. if includeFlags&IncludeModes != 0 {
  319. tx.Set(fmt.Sprintf(keyChannelPassword, channelKey), channelInfo.Key, nil)
  320. modeString := modes.Modes(channelInfo.Modes).String()
  321. tx.Set(fmt.Sprintf(keyChannelModes, channelKey), modeString, nil)
  322. tx.Set(fmt.Sprintf(keyChannelUserLimit, channelKey), strconv.Itoa(channelInfo.UserLimit), nil)
  323. }
  324. if includeFlags&IncludeLists != 0 {
  325. banlistString, _ := json.Marshal(channelInfo.Bans)
  326. tx.Set(fmt.Sprintf(keyChannelBanlist, channelKey), string(banlistString), nil)
  327. exceptlistString, _ := json.Marshal(channelInfo.Excepts)
  328. tx.Set(fmt.Sprintf(keyChannelExceptlist, channelKey), string(exceptlistString), nil)
  329. invitelistString, _ := json.Marshal(channelInfo.Invites)
  330. tx.Set(fmt.Sprintf(keyChannelInvitelist, channelKey), string(invitelistString), nil)
  331. accountToUModeString, _ := json.Marshal(channelInfo.AccountToUMode)
  332. tx.Set(fmt.Sprintf(keyChannelAccountToUMode, channelKey), string(accountToUModeString), nil)
  333. }
  334. if includeFlags&IncludeSettings != 0 {
  335. settingsString, _ := json.Marshal(channelInfo.Settings)
  336. tx.Set(fmt.Sprintf(keyChannelSettings, channelKey), string(settingsString), nil)
  337. }
  338. }
  339. // PurgeChannel records a channel purge.
  340. func (reg *ChannelRegistry) PurgeChannel(chname string, record ChannelPurgeRecord) (err error) {
  341. serialized, err := json.Marshal(record)
  342. if err != nil {
  343. return err
  344. }
  345. serializedStr := string(serialized)
  346. key := fmt.Sprintf(keyChannelPurged, chname)
  347. return reg.server.store.Update(func(tx *buntdb.Tx) error {
  348. tx.Set(key, serializedStr, nil)
  349. return nil
  350. })
  351. }
  352. // LoadPurgeRecord retrieves information about whether and how a channel was purged.
  353. func (reg *ChannelRegistry) LoadPurgeRecord(chname string) (record ChannelPurgeRecord, err error) {
  354. var rawRecord string
  355. key := fmt.Sprintf(keyChannelPurged, chname)
  356. reg.server.store.View(func(tx *buntdb.Tx) error {
  357. rawRecord, _ = tx.Get(key)
  358. return nil
  359. })
  360. if rawRecord == "" {
  361. err = errNoSuchChannel
  362. return
  363. }
  364. err = json.Unmarshal([]byte(rawRecord), &record)
  365. if err != nil {
  366. reg.server.logger.Error("internal", "corrupt purge record", chname, err.Error())
  367. err = errNoSuchChannel
  368. return
  369. }
  370. return
  371. }
  372. // UnpurgeChannel deletes the record of a channel purge.
  373. func (reg *ChannelRegistry) UnpurgeChannel(chname string) (err error) {
  374. key := fmt.Sprintf(keyChannelPurged, chname)
  375. return reg.server.store.Update(func(tx *buntdb.Tx) error {
  376. tx.Delete(key)
  377. return nil
  378. })
  379. }