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.

chanserv.go 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. // Copyright (c) 2017 Daniel Oaks <daniel@danieloaks.net>
  2. // released under the MIT license
  3. package irc
  4. import (
  5. "bytes"
  6. "fmt"
  7. "hash/crc32"
  8. "sort"
  9. "strconv"
  10. "strings"
  11. "time"
  12. "github.com/goshuirc/irc-go/ircfmt"
  13. "github.com/oragono/oragono/irc/modes"
  14. "github.com/oragono/oragono/irc/sno"
  15. )
  16. const chanservHelp = `ChanServ lets you register and manage channels.`
  17. const chanservMask = "ChanServ!ChanServ@localhost"
  18. func chanregEnabled(config *Config) bool {
  19. return config.Channels.Registration.Enabled
  20. }
  21. var (
  22. chanservCommands = map[string]*serviceCommand{
  23. "op": {
  24. handler: csOpHandler,
  25. help: `Syntax: $bOP #channel [nickname]$b
  26. OP makes the given nickname, or yourself, a channel admin. You can only use
  27. this command if you're the founder of the channel.`,
  28. helpShort: `$bOP$b makes the given user (or yourself) a channel admin.`,
  29. authRequired: true,
  30. enabled: chanregEnabled,
  31. minParams: 1,
  32. },
  33. "register": {
  34. handler: csRegisterHandler,
  35. help: `Syntax: $bREGISTER #channel$b
  36. REGISTER lets you own the given channel. If you rejoin this channel, you'll be
  37. given admin privs on it. Modes set on the channel and the topic will also be
  38. remembered.`,
  39. helpShort: `$bREGISTER$b lets you own a given channel.`,
  40. authRequired: true,
  41. enabled: chanregEnabled,
  42. minParams: 1,
  43. },
  44. "unregister": {
  45. handler: csUnregisterHandler,
  46. help: `Syntax: $bUNREGISTER #channel [code]$b
  47. UNREGISTER deletes a channel registration, allowing someone else to claim it.
  48. To prevent accidental unregistrations, a verification code is required;
  49. invoking the command without a code will display the necessary code.`,
  50. helpShort: `$bUNREGISTER$b deletes a channel registration.`,
  51. enabled: chanregEnabled,
  52. minParams: 1,
  53. },
  54. "drop": {
  55. aliasOf: "unregister",
  56. },
  57. "amode": {
  58. handler: csAmodeHandler,
  59. help: `Syntax: $bAMODE #channel [mode change] [account]$b
  60. AMODE lists or modifies persistent mode settings that affect channel members.
  61. For example, $bAMODE #channel +o dan$b grants the the holder of the "dan"
  62. account the +o operator mode every time they join #channel. To list current
  63. accounts and modes, use $bAMODE #channel$b. Note that users are always
  64. referenced by their registered account names, not their nicknames.`,
  65. helpShort: `$bAMODE$b modifies persistent mode settings for channel members.`,
  66. enabled: chanregEnabled,
  67. minParams: 1,
  68. },
  69. "clear": {
  70. handler: csClearHandler,
  71. help: `Syntax: $bCLEAR #channel target$b
  72. CLEAR removes users or settings from a channel. Specifically:
  73. $bCLEAR #channel users$b kicks all users except for you.
  74. $bCLEAR #channel access$b resets all stored bans, invites, ban exceptions,
  75. and persistent user-mode grants made with CS AMODE.`,
  76. helpShort: `$bCLEAR$b removes users or settings from a channel.`,
  77. enabled: chanregEnabled,
  78. minParams: 2,
  79. },
  80. "transfer": {
  81. handler: csTransferHandler,
  82. help: `Syntax: $bTRANSFER [accept] #channel user [code]$b
  83. TRANSFER transfers ownership of a channel from one user to another.
  84. To prevent accidental transfers, a verification code is required. For
  85. example, $bTRANSFER #channel alice$b displays the required confirmation
  86. code, then $bTRANSFER #channel alice 2930242125$b initiates the transfer.
  87. Unless you are an IRC operator with the correct permissions, alice must
  88. then accept the transfer, which she can do with $bTRANSFER accept #channel$b.
  89. To cancel a pending transfer, transfer the channel to yourself.`,
  90. helpShort: `$bTRANSFER$b transfers ownership of a channel to another user.`,
  91. enabled: chanregEnabled,
  92. minParams: 2,
  93. },
  94. "purge": {
  95. handler: csPurgeHandler,
  96. help: `Syntax: $bPURGE #channel [reason]$b
  97. PURGE blacklists a channel from the server, making it impossible to join
  98. or otherwise interact with the channel. If the channel currently has members,
  99. they will be kicked from it. PURGE may also be applied preemptively to
  100. channels that do not currently have members.`,
  101. helpShort: `$bPURGE$b blacklists a channel from the server.`,
  102. capabs: []string{"chanreg"},
  103. minParams: 1,
  104. maxParams: 2,
  105. unsplitFinalParam: true,
  106. },
  107. "unpurge": {
  108. handler: csUnpurgeHandler,
  109. help: `Syntax: $bUNPURGE #channel$b
  110. UNPURGE removes any blacklisting of a channel that was previously
  111. set using PURGE.`,
  112. helpShort: `$bUNPURGE$b undoes a previous PURGE command.`,
  113. capabs: []string{"chanreg"},
  114. minParams: 1,
  115. },
  116. "info": {
  117. handler: csInfoHandler,
  118. help: `Syntax: $INFO #channel$b
  119. INFO displays info about a registered channel.`,
  120. helpShort: `$bINFO$b displays info about a registered channel.`,
  121. enabled: chanregEnabled,
  122. minParams: 1,
  123. },
  124. }
  125. )
  126. // csNotice sends the client a notice from ChanServ
  127. func csNotice(rb *ResponseBuffer, text string) {
  128. rb.Add(nil, chanservMask, "NOTICE", rb.target.Nick(), text)
  129. }
  130. func csAmodeHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  131. channelName := params[0]
  132. channel := server.channels.Get(channelName)
  133. if channel == nil {
  134. csNotice(rb, client.t("Channel does not exist"))
  135. return
  136. } else if channel.Founder() == "" {
  137. csNotice(rb, client.t("Channel is not registered"))
  138. return
  139. }
  140. modeChanges, unknown := modes.ParseChannelModeChanges(params[1:]...)
  141. var change modes.ModeChange
  142. if len(modeChanges) > 1 || len(unknown) > 0 {
  143. csNotice(rb, client.t("Invalid mode change"))
  144. return
  145. } else if len(modeChanges) == 1 {
  146. change = modeChanges[0]
  147. } else {
  148. change = modes.ModeChange{Op: modes.List}
  149. }
  150. // normalize and validate the account argument
  151. accountIsValid := false
  152. change.Arg, _ = CasefoldName(change.Arg)
  153. switch change.Op {
  154. case modes.List:
  155. accountIsValid = true
  156. case modes.Add:
  157. // if we're adding a mode, the account must exist
  158. if change.Arg != "" {
  159. _, err := server.accounts.LoadAccount(change.Arg)
  160. accountIsValid = (err == nil)
  161. }
  162. case modes.Remove:
  163. // allow removal of accounts that may have been deleted
  164. accountIsValid = (change.Arg != "")
  165. }
  166. if !accountIsValid {
  167. csNotice(rb, client.t("Account does not exist"))
  168. return
  169. }
  170. affectedModes, err := channel.ProcessAccountToUmodeChange(client, change)
  171. if err == errInsufficientPrivs {
  172. csNotice(rb, client.t("Insufficient privileges"))
  173. return
  174. } else if err != nil {
  175. csNotice(rb, client.t("Internal error"))
  176. return
  177. }
  178. switch change.Op {
  179. case modes.List:
  180. // sort the persistent modes in descending order of priority
  181. sort.Slice(affectedModes, func(i, j int) bool {
  182. return umodeGreaterThan(affectedModes[i].Mode, affectedModes[j].Mode)
  183. })
  184. csNotice(rb, fmt.Sprintf(client.t("Channel %[1]s has %[2]d persistent modes set"), channelName, len(affectedModes)))
  185. for _, modeChange := range affectedModes {
  186. csNotice(rb, fmt.Sprintf(client.t("Account %[1]s receives mode +%[2]s"), modeChange.Arg, string(modeChange.Mode)))
  187. }
  188. case modes.Add, modes.Remove:
  189. if len(affectedModes) > 0 {
  190. csNotice(rb, fmt.Sprintf(client.t("Successfully set mode %s"), change.String()))
  191. } else {
  192. csNotice(rb, client.t("No changes were made"))
  193. }
  194. }
  195. }
  196. func csOpHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  197. channelInfo := server.channels.Get(params[0])
  198. if channelInfo == nil {
  199. csNotice(rb, client.t("Channel does not exist"))
  200. return
  201. }
  202. channelName := channelInfo.Name()
  203. clientAccount := client.Account()
  204. if clientAccount == "" || clientAccount != channelInfo.Founder() {
  205. csNotice(rb, client.t("Only the channel founder can do this"))
  206. return
  207. }
  208. var target *Client
  209. if len(params) > 1 {
  210. target = server.clients.Get(params[1])
  211. if target == nil {
  212. csNotice(rb, client.t("Could not find given client"))
  213. return
  214. }
  215. } else {
  216. target = client
  217. }
  218. // give them privs
  219. givenMode := modes.ChannelOperator
  220. if clientAccount == target.Account() {
  221. givenMode = modes.ChannelFounder
  222. }
  223. change := channelInfo.applyModeToMember(client, givenMode, modes.Add, target.NickCasefolded(), rb)
  224. if change != nil {
  225. //TODO(dan): we should change the name of String and make it return a slice here
  226. //TODO(dan): unify this code with code in modes.go
  227. args := append([]string{channelName}, strings.Split(change.String(), " ")...)
  228. for _, member := range channelInfo.Members() {
  229. member.Send(nil, fmt.Sprintf("ChanServ!services@%s", client.server.name), "MODE", args...)
  230. }
  231. }
  232. csNotice(rb, fmt.Sprintf(client.t("Successfully op'd in channel %s"), channelName))
  233. tnick := target.Nick()
  234. server.logger.Info("services", fmt.Sprintf("Client %s op'd [%s] in channel %s", client.Nick(), tnick, channelName))
  235. server.snomasks.Send(sno.LocalChannels, fmt.Sprintf(ircfmt.Unescape("Client $c[grey][$r%s$c[grey]] CS OP'd $c[grey][$r%s$c[grey]] in channel $c[grey][$r%s$c[grey]]"), client.NickMaskString(), tnick, channelName))
  236. }
  237. func csRegisterHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  238. channelName := params[0]
  239. channelKey, err := CasefoldChannel(channelName)
  240. if err != nil {
  241. csNotice(rb, client.t("Channel name is not valid"))
  242. return
  243. }
  244. channelInfo := server.channels.Get(channelKey)
  245. if channelInfo == nil || !channelInfo.ClientIsAtLeast(client, modes.ChannelOperator) {
  246. csNotice(rb, client.t("You must be an oper on the channel to register it"))
  247. return
  248. }
  249. account := client.Account()
  250. if !checkChanLimit(client, rb) {
  251. return
  252. }
  253. // this provides the synchronization that allows exactly one registration of the channel:
  254. err = server.channels.SetRegistered(channelKey, account)
  255. if err != nil {
  256. csNotice(rb, err.Error())
  257. return
  258. }
  259. csNotice(rb, fmt.Sprintf(client.t("Channel %s successfully registered"), channelName))
  260. server.logger.Info("services", fmt.Sprintf("Client %s registered channel %s", client.Nick(), channelName))
  261. server.snomasks.Send(sno.LocalChannels, fmt.Sprintf(ircfmt.Unescape("Channel registered $c[grey][$r%s$c[grey]] by $c[grey][$r%s$c[grey]]"), channelName, client.nickMaskString))
  262. // give them founder privs
  263. change := channelInfo.applyModeToMember(client, modes.ChannelFounder, modes.Add, client.NickCasefolded(), rb)
  264. if change != nil {
  265. //TODO(dan): we should change the name of String and make it return a slice here
  266. //TODO(dan): unify this code with code in modes.go
  267. args := append([]string{channelName}, strings.Split(change.String(), " ")...)
  268. for _, member := range channelInfo.Members() {
  269. member.Send(nil, fmt.Sprintf("ChanServ!services@%s", client.server.name), "MODE", args...)
  270. }
  271. }
  272. }
  273. // check whether a client has already registered too many channels
  274. func checkChanLimit(client *Client, rb *ResponseBuffer) (ok bool) {
  275. account := client.Account()
  276. channelsAlreadyRegistered := client.server.accounts.ChannelsForAccount(account)
  277. ok = len(channelsAlreadyRegistered) < client.server.Config().Channels.Registration.MaxChannelsPerAccount || client.HasRoleCapabs("chanreg")
  278. if !ok {
  279. csNotice(rb, client.t("You have already registered the maximum number of channels; try dropping some with /CS UNREGISTER"))
  280. }
  281. return
  282. }
  283. func csUnregisterHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  284. channelName := params[0]
  285. var verificationCode string
  286. if len(params) > 1 {
  287. verificationCode = params[1]
  288. }
  289. channelKey, err := CasefoldChannel(channelName)
  290. if channelKey == "" || err != nil {
  291. csNotice(rb, client.t("Channel name is not valid"))
  292. return
  293. }
  294. channel := server.channels.Get(channelKey)
  295. if channel == nil {
  296. csNotice(rb, client.t("No such channel"))
  297. return
  298. }
  299. founder := channel.Founder()
  300. if founder == "" {
  301. csNotice(rb, client.t("That channel is not registered"))
  302. return
  303. }
  304. hasPrivs := client.HasRoleCapabs("chanreg") || founder == client.Account()
  305. if !hasPrivs {
  306. csNotice(rb, client.t("Insufficient privileges"))
  307. return
  308. }
  309. info := channel.ExportRegistration(0)
  310. expectedCode := unregisterConfirmationCode(info.Name, info.RegisteredAt)
  311. if expectedCode != verificationCode {
  312. csNotice(rb, ircfmt.Unescape(client.t("$bWarning: unregistering this channel will remove all stored channel attributes.$b")))
  313. csNotice(rb, fmt.Sprintf(client.t("To confirm channel unregistration, type: /CS UNREGISTER %[1]s %[2]s"), channelKey, expectedCode))
  314. return
  315. }
  316. server.channels.SetUnregistered(channelKey, founder)
  317. csNotice(rb, fmt.Sprintf(client.t("Channel %s is now unregistered"), channelKey))
  318. }
  319. // deterministically generates a confirmation code for unregistering a channel / account
  320. func unregisterConfirmationCode(name string, registeredAt time.Time) (code string) {
  321. var codeInput bytes.Buffer
  322. codeInput.WriteString(name)
  323. codeInput.WriteString(strconv.FormatInt(registeredAt.Unix(), 16))
  324. return strconv.Itoa(int(crc32.ChecksumIEEE(codeInput.Bytes())))
  325. }
  326. func csClearHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  327. channel := server.channels.Get(params[0])
  328. if channel == nil {
  329. csNotice(rb, client.t("Channel does not exist"))
  330. return
  331. }
  332. account := client.Account()
  333. if !(client.HasRoleCapabs("chanreg") || (account != "" && account == channel.Founder())) {
  334. csNotice(rb, client.t("Insufficient privileges"))
  335. return
  336. }
  337. switch strings.ToLower(params[1]) {
  338. case "access":
  339. channel.resetAccess()
  340. csNotice(rb, client.t("Successfully reset channel access"))
  341. case "users":
  342. for _, target := range channel.Members() {
  343. if target != client {
  344. channel.Kick(client, target, "Cleared by ChanServ", rb, true)
  345. }
  346. }
  347. default:
  348. csNotice(rb, client.t("Invalid parameters"))
  349. }
  350. }
  351. func csTransferHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  352. if strings.ToLower(params[0]) == "accept" {
  353. processTransferAccept(client, params[1], rb)
  354. return
  355. }
  356. chname := params[0]
  357. channel := server.channels.Get(chname)
  358. if channel == nil {
  359. csNotice(rb, client.t("Channel does not exist"))
  360. return
  361. }
  362. regInfo := channel.ExportRegistration(0)
  363. chname = regInfo.Name
  364. account := client.Account()
  365. isFounder := account != "" && account == regInfo.Founder
  366. hasPrivs := client.HasRoleCapabs("chanreg")
  367. if !(isFounder || hasPrivs) {
  368. csNotice(rb, client.t("Insufficient privileges"))
  369. return
  370. }
  371. target := params[1]
  372. targetAccount, err := server.accounts.LoadAccount(params[1])
  373. if err != nil {
  374. csNotice(rb, client.t("Account does not exist"))
  375. return
  376. }
  377. if targetAccount.NameCasefolded != account {
  378. expectedCode := unregisterConfirmationCode(regInfo.Name, regInfo.RegisteredAt)
  379. codeValidated := 2 < len(params) && params[2] == expectedCode
  380. if !codeValidated {
  381. csNotice(rb, ircfmt.Unescape(client.t("$bWarning: you are about to transfer control of your channel to another user.$b")))
  382. csNotice(rb, fmt.Sprintf(client.t("To confirm your channel transfer, type: /CS TRANSFER %[1]s %[2]s %[3]s"), chname, target, expectedCode))
  383. return
  384. }
  385. }
  386. status, err := channel.Transfer(client, target, hasPrivs)
  387. if err == nil {
  388. switch status {
  389. case channelTransferComplete:
  390. csNotice(rb, fmt.Sprintf(client.t("Successfully transferred channel %[1]s to account %[2]s"), chname, target))
  391. case channelTransferPending:
  392. sendTransferPendingNotice(server, target, chname)
  393. csNotice(rb, fmt.Sprintf(client.t("Transfer of channel %[1]s to account %[2]s succeeded, pending acceptance"), chname, target))
  394. case channelTransferCancelled:
  395. csNotice(rb, fmt.Sprintf(client.t("Cancelled pending transfer of channel %s"), chname))
  396. }
  397. } else {
  398. csNotice(rb, client.t("Could not transfer channel"))
  399. }
  400. }
  401. func sendTransferPendingNotice(server *Server, account, chname string) {
  402. clients := server.accounts.AccountToClients(account)
  403. if len(clients) == 0 {
  404. return
  405. }
  406. var client *Client
  407. for _, candidate := range clients {
  408. client = candidate
  409. if candidate.NickCasefolded() == candidate.Account() {
  410. break // prefer the login where the nick is the account
  411. }
  412. }
  413. client.Send(nil, chanservMask, "NOTICE", client.Nick(), fmt.Sprintf(client.t("You have been offered ownership of channel %[1]s. To accept, /CS TRANSFER ACCEPT %[1]s"), chname))
  414. }
  415. func processTransferAccept(client *Client, chname string, rb *ResponseBuffer) {
  416. channel := client.server.channels.Get(chname)
  417. if channel == nil {
  418. csNotice(rb, client.t("Channel does not exist"))
  419. return
  420. }
  421. if !checkChanLimit(client, rb) {
  422. return
  423. }
  424. switch channel.AcceptTransfer(client) {
  425. case nil:
  426. csNotice(rb, fmt.Sprintf(client.t("Successfully accepted ownership of channel %s"), channel.Name()))
  427. case errChannelTransferNotOffered:
  428. csNotice(rb, fmt.Sprintf(client.t("You weren't offered ownership of channel %s"), channel.Name()))
  429. default:
  430. csNotice(rb, fmt.Sprintf(client.t("Could not accept ownership of channel %s"), channel.Name()))
  431. }
  432. }
  433. func csPurgeHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  434. oper := client.Oper()
  435. if oper == nil {
  436. return // should be impossible because you need oper capabs for this
  437. }
  438. chname := params[0]
  439. var reason string
  440. if 1 < len(params) {
  441. reason = params[1]
  442. }
  443. purgeRecord := ChannelPurgeRecord{
  444. Oper: oper.Name,
  445. PurgedAt: time.Now().UTC(),
  446. Reason: reason,
  447. }
  448. switch server.channels.Purge(chname, purgeRecord) {
  449. case nil:
  450. channel := server.channels.Get(chname)
  451. if channel != nil { // channel need not exist to be purged
  452. for _, target := range channel.Members() {
  453. channel.Kick(client, target, "Cleared by ChanServ", rb, true)
  454. }
  455. }
  456. csNotice(rb, fmt.Sprintf(client.t("Successfully purged channel %s from the server"), chname))
  457. case errInvalidChannelName:
  458. csNotice(rb, fmt.Sprintf(client.t("Can't purge invalid channel %s"), chname))
  459. default:
  460. csNotice(rb, client.t("An error occurred"))
  461. }
  462. }
  463. func csUnpurgeHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  464. chname := params[0]
  465. switch server.channels.Unpurge(chname) {
  466. case nil:
  467. csNotice(rb, fmt.Sprintf(client.t("Successfully unpurged channel %s from the server"), chname))
  468. case errNoSuchChannel:
  469. csNotice(rb, fmt.Sprintf(client.t("Channel %s wasn't previously purged from the server"), chname))
  470. default:
  471. csNotice(rb, client.t("An error occurred"))
  472. }
  473. }
  474. func csInfoHandler(server *Server, client *Client, command string, params []string, rb *ResponseBuffer) {
  475. chname, err := CasefoldChannel(params[0])
  476. if err != nil {
  477. csNotice(rb, client.t("Invalid channel name"))
  478. return
  479. }
  480. // purge status
  481. if client.HasRoleCapabs("chanreg") {
  482. purgeRecord, err := server.channelRegistry.LoadPurgeRecord(chname)
  483. if err == nil {
  484. csNotice(rb, fmt.Sprintf(client.t("Channel %s was purged by the server operators and cannot be used"), chname))
  485. csNotice(rb, fmt.Sprintf(client.t("Purged by operator: %s"), purgeRecord.Oper))
  486. csNotice(rb, fmt.Sprintf(client.t("Purged at: %s"), purgeRecord.PurgedAt.Format(time.RFC1123)))
  487. if purgeRecord.Reason != "" {
  488. csNotice(rb, fmt.Sprintf(client.t("Purge reason: %s"), purgeRecord.Reason))
  489. }
  490. }
  491. } else {
  492. if server.channels.IsPurged(chname) {
  493. csNotice(rb, fmt.Sprintf(client.t("Channel %s was purged by the server operators and cannot be used"), chname))
  494. }
  495. }
  496. var chinfo RegisteredChannel
  497. channel := server.channels.Get(params[0])
  498. if channel != nil {
  499. chinfo = channel.ExportRegistration(0)
  500. } else {
  501. chinfo, err = server.channelRegistry.LoadChannel(chname)
  502. if err != nil && !(err == errNoSuchChannel || err == errFeatureDisabled) {
  503. csNotice(rb, client.t("An error occurred"))
  504. return
  505. }
  506. }
  507. // channel exists but is unregistered, or doesn't exist:
  508. if chinfo.Founder == "" {
  509. csNotice(rb, fmt.Sprintf(client.t("Channel %s is not registered"), chname))
  510. return
  511. }
  512. csNotice(rb, fmt.Sprintf(client.t("Channel %s is registered"), chinfo.Name))
  513. csNotice(rb, fmt.Sprintf(client.t("Founder: %s"), chinfo.Founder))
  514. csNotice(rb, fmt.Sprintf(client.t("Registered at: %s"), chinfo.RegisteredAt.Format(time.RFC1123)))
  515. }