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.

ChannelStateHandlerTest.kt 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. package com.dmdirc.ktirc.events.handlers
  2. import com.dmdirc.ktirc.BehaviourConfig
  3. import com.dmdirc.ktirc.IrcClient
  4. import com.dmdirc.ktirc.TestConstants
  5. import com.dmdirc.ktirc.events.*
  6. import com.dmdirc.ktirc.io.CaseMapping
  7. import com.dmdirc.ktirc.model.*
  8. import io.mockk.every
  9. import io.mockk.mockk
  10. import io.mockk.verify
  11. import org.junit.jupiter.api.Assertions.*
  12. import org.junit.jupiter.api.Test
  13. internal class ChannelStateHandlerTest {
  14. private val handler = ChannelStateHandler()
  15. private val fakeChannelState = ChannelStateMap { CaseMapping.Rfc }
  16. private val fakeServerState = ServerState("", "")
  17. private val behaviourConfig = BehaviourConfig()
  18. private val ircClient = mockk<IrcClient> {
  19. every { serverState } returns fakeServerState
  20. every { channelState } returns fakeChannelState
  21. every { behaviour } returns behaviourConfig
  22. every { isLocalUser(any<User>()) } answers { arg<User>(0) == User("acidburn", "libby", "root.localhost") }
  23. every { isLocalUser(any<String>()) } answers { arg<String>(0) == "acidburn" }
  24. }
  25. @Test
  26. fun `creates new state object for local joins`() {
  27. handler.processEvent(ircClient, ChannelJoined(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson"))
  28. assertTrue("#thegibson" in fakeChannelState)
  29. }
  30. @Test
  31. fun `does not create new state object for remote joins`() {
  32. handler.processEvent(ircClient, ChannelJoined(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  33. assertFalse("#thegibson" in fakeChannelState)
  34. }
  35. @Test
  36. fun `adds joiners to channel state`() {
  37. fakeChannelState += ChannelState("#thegibson") { CaseMapping.Rfc }
  38. handler.processEvent(ircClient, ChannelJoined(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  39. assertTrue("zerocool" in fakeChannelState["#thegibson"]?.users!!)
  40. }
  41. @Test
  42. fun `ignores duplicate joiners`() {
  43. fakeChannelState += ChannelState("#thegibson") { CaseMapping.Rfc }
  44. handler.processEvent(ircClient, ChannelJoined(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  45. handler.processEvent(ircClient, ChannelJoined(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  46. assertTrue("zerocool" in fakeChannelState["#thegibson"]?.users!!)
  47. assertEquals(1, fakeChannelState["#thegibson"]?.users?.count())
  48. }
  49. @Test
  50. fun `clears existing users when getting a new list`() {
  51. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  52. channel.users += ChannelUser("acidBurn")
  53. channel.users += ChannelUser("thePlague")
  54. fakeChannelState += channel
  55. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("zeroCool")))
  56. assertEquals(1, channel.users.count())
  57. assertNotNull(channel.users["zeroCool"])
  58. }
  59. @Test
  60. fun `adds users from multiple name received events`() {
  61. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  62. fakeChannelState += channel
  63. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("zeroCool")))
  64. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("acidBurn")))
  65. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("thePlague")))
  66. assertEquals(3, channel.users.count())
  67. assertNotNull(channel.users["zeroCool"])
  68. assertNotNull(channel.users["acidBurn"])
  69. assertNotNull(channel.users["thePlague"])
  70. }
  71. @Test
  72. fun `clears and readds users on additional names received`() {
  73. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  74. fakeChannelState += channel
  75. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("zeroCool")))
  76. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  77. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("acidBurn")))
  78. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("thePlague")))
  79. assertEquals(2, channel.users.count())
  80. assertNotNull(channel.users["acidBurn"])
  81. assertNotNull(channel.users["thePlague"])
  82. }
  83. @Test
  84. fun `adds users with mode prefixes`() {
  85. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  86. fakeChannelState += channel
  87. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  88. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool", "@+acidBurn", "+thePlague", "cerealKiller")))
  89. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  90. assertEquals(4, channel.users.count())
  91. assertEquals("o", channel.users["zeroCool"]?.modes)
  92. assertEquals("ov", channel.users["acidBurn"]?.modes)
  93. assertEquals("v", channel.users["thePlague"]?.modes)
  94. assertEquals("", channel.users["cerealKiller"]?.modes)
  95. }
  96. @Test
  97. fun `adds users with full hosts`() {
  98. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  99. fakeChannelState += channel
  100. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  101. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool!dade@root.localhost", "+acidBurn!libby@root.localhost")))
  102. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  103. assertEquals(2, channel.users.count())
  104. assertEquals("o", channel.users["zeroCool"]?.modes)
  105. assertEquals("v", channel.users["acidBurn"]?.modes)
  106. }
  107. @Test
  108. fun `updates receiving user list state`() {
  109. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  110. fakeChannelState += channel
  111. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  112. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool!dade@root.localhost", "+acidBurn!libby@root.localhost")))
  113. assertTrue(channel.receivingUserList)
  114. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  115. assertFalse(channel.receivingUserList)
  116. }
  117. @Test
  118. fun `requests modes on end of names if configured and undiscovered`() {
  119. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  120. fakeChannelState += channel
  121. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  122. behaviourConfig.requestModesOnJoin = true
  123. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  124. verify {
  125. ircClient.send("MODE", "#thegibson")
  126. }
  127. }
  128. @Test
  129. fun `does not request modes on end of names if already discovered`() {
  130. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  131. fakeChannelState += channel
  132. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  133. behaviourConfig.requestModesOnJoin = true
  134. channel.modesDiscovered = true
  135. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  136. verify(inverse = true) {
  137. ircClient.send("MODE", "#thegibson")
  138. }
  139. }
  140. @Test
  141. fun `does not request modes on end of names if not configured`() {
  142. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  143. fakeChannelState += channel
  144. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  145. behaviourConfig.requestModesOnJoin = false
  146. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  147. verify(inverse = true) {
  148. ircClient.send("MODE", "#thegibson")
  149. }
  150. }
  151. @Test
  152. fun `removes state object for local parts`() {
  153. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  154. fakeChannelState += channel
  155. handler.processEvent(ircClient, ChannelParted(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson"))
  156. assertFalse("#thegibson" in fakeChannelState)
  157. }
  158. @Test
  159. fun `removes user from channel member list for remote parts`() {
  160. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  161. channel.users += ChannelUser("ZeroCool")
  162. fakeChannelState += channel
  163. handler.processEvent(ircClient, ChannelParted(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  164. assertFalse("zerocool" in channel.users)
  165. }
  166. @Test
  167. fun `removes state object for local kicks`() {
  168. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  169. fakeChannelState += channel
  170. handler.processEvent(ircClient, ChannelUserKicked(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson", "acidburn", "Bye!"))
  171. assertFalse("#thegibson" in fakeChannelState)
  172. }
  173. @Test
  174. fun `removes user from channel member list for remote kicks`() {
  175. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  176. channel.users += ChannelUser("ZeroCool")
  177. fakeChannelState += channel
  178. handler.processEvent(ircClient, ChannelUserKicked(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson", "zerocool", "Bye!"))
  179. assertFalse("zerocool" in channel.users)
  180. }
  181. @Test
  182. fun `removes user from channel member lists for quits`() {
  183. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  184. users += ChannelUser("ZeroCool")
  185. fakeChannelState += this
  186. }
  187. with (ChannelState("#dumpsterdiving") { CaseMapping.Rfc }) {
  188. users += ChannelUser("ZeroCool")
  189. fakeChannelState += this
  190. }
  191. with (ChannelState("#chat") { CaseMapping.Rfc }) {
  192. users += ChannelUser("AcidBurn")
  193. fakeChannelState += this
  194. }
  195. handler.processEvent(ircClient, ChannelQuit(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  196. handler.processEvent(ircClient, ChannelQuit(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#dumpsterdiving"))
  197. assertFalse("zerocool" in fakeChannelState["#thegibson"]!!.users)
  198. assertFalse("zerocool" in fakeChannelState["#dumpsterdiving"]!!.users)
  199. assertFalse("zerocool" in fakeChannelState["#chat"]!!.users)
  200. assertTrue("acidburn" in fakeChannelState["#chat"]!!.users)
  201. }
  202. @Test
  203. fun `renames user in channel member list for nick changes`() {
  204. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  205. channel.users += ChannelUser("acidBurn")
  206. fakeChannelState += channel
  207. handler.processEvent(ircClient, ChannelNickChanged(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson", "acidB"))
  208. handler.processEvent(ircClient, ChannelNickChanged(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#dumpsterdiving", "acidB"))
  209. assertFalse("acidBurn" in channel.users)
  210. assertTrue("acidB" in channel.users)
  211. assertEquals("acidB", channel.users["acidB"]?.nickname)
  212. }
  213. @Test
  214. fun `sets mode discovered flag when discovered mode event received`() {
  215. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  216. fakeChannelState += channel
  217. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  218. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+", emptyArray(), true))
  219. assertTrue(channel.modesDiscovered)
  220. }
  221. @Test
  222. fun `adds modes when discovered mode event received`() {
  223. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  224. fakeChannelState += channel
  225. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  226. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+ceg", arrayOf("CCC", "EEE"), true))
  227. assertEquals("CCC", channel.modes['c'])
  228. assertEquals("EEE", channel.modes['e'])
  229. assertEquals("", channel.modes['g'])
  230. }
  231. @Test
  232. fun `adjusts complex modes when mode change event received`() {
  233. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  234. channel.modes['c'] = "CCC"
  235. channel.modes['e'] = "EEE"
  236. channel.modes['h'] = ""
  237. fakeChannelState += channel
  238. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  239. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "-c+d-eh+fg", arrayOf("CCC", "DDD", "FFF"), true))
  240. assertNull(channel.modes['c'])
  241. assertEquals("DDD", channel.modes['d'])
  242. assertNull(channel.modes['e'])
  243. assertEquals("FFF", channel.modes['f'])
  244. assertEquals("", channel.modes['g'])
  245. assertNull(channel.modes['h'])
  246. }
  247. @Test
  248. fun `handles unprivileged user gaining new mode`() {
  249. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  250. users += ChannelUser("ZeroCool")
  251. fakeChannelState += this
  252. }
  253. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+o", arrayOf("zeroCool")))
  254. assertEquals("o", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  255. }
  256. @Test
  257. fun `handles privileged user gaining lesser mode`() {
  258. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  259. users += ChannelUser("ZeroCool", "o")
  260. fakeChannelState += this
  261. }
  262. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+v", arrayOf("zeroCool")))
  263. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  264. }
  265. @Test
  266. fun `handles privileged user gaining greater mode`() {
  267. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  268. users += ChannelUser("ZeroCool", "v")
  269. fakeChannelState += this
  270. }
  271. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+o", arrayOf("zeroCool")))
  272. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  273. }
  274. @Test
  275. fun `handles user gaining multiple modes`() {
  276. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  277. users += ChannelUser("ZeroCool")
  278. fakeChannelState += this
  279. }
  280. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+vo", arrayOf("zeroCool", "zeroCool")))
  281. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  282. }
  283. @Test
  284. fun `handles user losing multiple modes`() {
  285. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  286. users += ChannelUser("ZeroCool", "ov")
  287. fakeChannelState += this
  288. }
  289. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "-vo", arrayOf("zeroCool", "zeroCool")))
  290. assertEquals("", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  291. }
  292. @Test
  293. fun `handles mixture of user modes and normal modes`() {
  294. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  295. users += ChannelUser("ZeroCool", "v")
  296. fakeChannelState += this
  297. }
  298. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  299. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "oa-v+b", arrayOf("zeroCool", "aaa", "zeroCool", "bbb")))
  300. assertEquals("o", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  301. assertEquals("aaa", fakeChannelState["#thegibson"]?.modes?.get('a'))
  302. assertEquals("bbb", fakeChannelState["#thegibson"]?.modes?.get('b'))
  303. }
  304. @Test
  305. fun `updates topic state when it's discovered for the first time`() {
  306. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  307. fakeChannelState += state
  308. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", "Hack the planet!"))
  309. handler.processEvent(ircClient, ChannelTopicMetadataDiscovered(EventMetadata(TestConstants.time), "#thegibson", User("acidBurn"), TestConstants.otherTime))
  310. assertTrue(state.topicDiscovered)
  311. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime), state.topic)
  312. }
  313. @Test
  314. fun `updates topic state when no topic is discovered for the first time`() {
  315. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  316. fakeChannelState += state
  317. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", null))
  318. assertTrue(state.topicDiscovered)
  319. assertEquals(ChannelTopic(), state.topic)
  320. }
  321. @Test
  322. fun `leaves topic state when it's discovered for a second time`() {
  323. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  324. state.topic = ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime)
  325. state.topicDiscovered = true
  326. fakeChannelState += state
  327. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", "Hack the planet"))
  328. handler.processEvent(ircClient, ChannelTopicMetadataDiscovered(EventMetadata(TestConstants.time), "#thegibson", User("zeroCool"), TestConstants.time))
  329. assertTrue(state.topicDiscovered)
  330. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime), state.topic)
  331. }
  332. @Test
  333. fun `updates topic state when the topic is changed`() {
  334. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  335. fakeChannelState += state
  336. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#thegibson", "Hack the planet!"))
  337. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.time), state.topic)
  338. }
  339. @Test
  340. fun `updates topic state when the topic is unset`() {
  341. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  342. fakeChannelState += state
  343. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#thegibson", null))
  344. assertEquals(ChannelTopic(null, User("acidBurn"), TestConstants.time), state.topic)
  345. }
  346. @Test
  347. fun `ignores topic change when channel doesn't exist`() {
  348. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  349. fakeChannelState += state
  350. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#dumpsterdiving", "Hack the planet!"))
  351. assertEquals(ChannelTopic(), state.topic)
  352. }
  353. }