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

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  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 `updates replacedUsers property of names finished event`() {
  85. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  86. fakeChannelState += channel
  87. val event = ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson")
  88. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("zeroCool")))
  89. handler.processEvent(ircClient, event)
  90. assertEquals(1, event.replacedUsers?.size)
  91. assertEquals("zeroCool", event.replacedUsers?.get(0))
  92. }
  93. @Test
  94. fun `adds users with mode prefixes`() {
  95. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  96. fakeChannelState += channel
  97. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  98. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool", "@+acidBurn", "+thePlague", "cerealKiller")))
  99. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  100. assertEquals(4, channel.users.count())
  101. assertEquals("o", channel.users["zeroCool"]?.modes)
  102. assertEquals("ov", channel.users["acidBurn"]?.modes)
  103. assertEquals("v", channel.users["thePlague"]?.modes)
  104. assertEquals("", channel.users["cerealKiller"]?.modes)
  105. }
  106. @Test
  107. fun `adds users with full hosts`() {
  108. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  109. fakeChannelState += channel
  110. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  111. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool!dade@root.localhost", "+acidBurn!libby@root.localhost")))
  112. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  113. assertEquals(2, channel.users.count())
  114. assertEquals("o", channel.users["zeroCool"]?.modes)
  115. assertEquals("v", channel.users["acidBurn"]?.modes)
  116. }
  117. @Test
  118. fun `updates receiving user list state`() {
  119. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  120. fakeChannelState += channel
  121. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  122. handler.processEvent(ircClient, ChannelNamesReceived(EventMetadata(TestConstants.time), "#thegibson", listOf("@zeroCool!dade@root.localhost", "+acidBurn!libby@root.localhost")))
  123. assertTrue(channel.receivingUserList)
  124. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  125. assertFalse(channel.receivingUserList)
  126. }
  127. @Test
  128. fun `requests modes on end of names if configured and undiscovered`() {
  129. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  130. fakeChannelState += channel
  131. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  132. behaviourConfig.requestModesOnJoin = true
  133. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  134. verify {
  135. ircClient.send("MODE", "#thegibson")
  136. }
  137. }
  138. @Test
  139. fun `does not request modes on end of names if already discovered`() {
  140. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  141. fakeChannelState += channel
  142. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  143. behaviourConfig.requestModesOnJoin = true
  144. channel.modesDiscovered = true
  145. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  146. verify(inverse = true) {
  147. ircClient.send("MODE", "#thegibson")
  148. }
  149. }
  150. @Test
  151. fun `does not request modes on end of names if not configured`() {
  152. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  153. fakeChannelState += channel
  154. fakeServerState.features[ServerFeature.ModePrefixes] = ModePrefixMapping("ov", "@+")
  155. behaviourConfig.requestModesOnJoin = false
  156. handler.processEvent(ircClient, ChannelNamesFinished(EventMetadata(TestConstants.time), "#thegibson"))
  157. verify(inverse = true) {
  158. ircClient.send("MODE", "#thegibson")
  159. }
  160. }
  161. @Test
  162. fun `removes state object for local parts`() {
  163. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  164. fakeChannelState += channel
  165. handler.processEvent(ircClient, ChannelParted(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson"))
  166. assertFalse("#thegibson" in fakeChannelState)
  167. }
  168. @Test
  169. fun `removes user from channel member list for remote parts`() {
  170. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  171. channel.users += ChannelUser("ZeroCool")
  172. fakeChannelState += channel
  173. handler.processEvent(ircClient, ChannelParted(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  174. assertFalse("zerocool" in channel.users)
  175. }
  176. @Test
  177. fun `removes state object for local kicks`() {
  178. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  179. fakeChannelState += channel
  180. handler.processEvent(ircClient, ChannelUserKicked(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson", "acidburn", "Bye!"))
  181. assertFalse("#thegibson" in fakeChannelState)
  182. }
  183. @Test
  184. fun `removes user from channel member list for remote kicks`() {
  185. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  186. channel.users += ChannelUser("ZeroCool")
  187. fakeChannelState += channel
  188. handler.processEvent(ircClient, ChannelUserKicked(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson", "zerocool", "Bye!"))
  189. assertFalse("zerocool" in channel.users)
  190. }
  191. @Test
  192. fun `removes user from channel member lists for quits`() {
  193. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  194. users += ChannelUser("ZeroCool")
  195. fakeChannelState += this
  196. }
  197. with (ChannelState("#dumpsterdiving") { CaseMapping.Rfc }) {
  198. users += ChannelUser("ZeroCool")
  199. fakeChannelState += this
  200. }
  201. with (ChannelState("#chat") { CaseMapping.Rfc }) {
  202. users += ChannelUser("AcidBurn")
  203. fakeChannelState += this
  204. }
  205. handler.processEvent(ircClient, ChannelQuit(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#thegibson"))
  206. handler.processEvent(ircClient, ChannelQuit(EventMetadata(TestConstants.time), User("zerocool", "dade", "root.localhost"), "#dumpsterdiving"))
  207. assertFalse("zerocool" in fakeChannelState["#thegibson"]!!.users)
  208. assertFalse("zerocool" in fakeChannelState["#dumpsterdiving"]!!.users)
  209. assertFalse("zerocool" in fakeChannelState["#chat"]!!.users)
  210. assertTrue("acidburn" in fakeChannelState["#chat"]!!.users)
  211. }
  212. @Test
  213. fun `renames user in channel member list for nick changes`() {
  214. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  215. channel.users += ChannelUser("acidBurn")
  216. fakeChannelState += channel
  217. handler.processEvent(ircClient, ChannelNickChanged(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#thegibson", "acidB"))
  218. handler.processEvent(ircClient, ChannelNickChanged(EventMetadata(TestConstants.time), User("acidburn", "libby", "root.localhost"), "#dumpsterdiving", "acidB"))
  219. assertFalse("acidBurn" in channel.users)
  220. assertTrue("acidB" in channel.users)
  221. assertEquals("acidB", channel.users["acidB"]?.nickname)
  222. }
  223. @Test
  224. fun `sets mode discovered flag when discovered mode event received`() {
  225. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  226. fakeChannelState += channel
  227. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  228. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+", emptyArray(), true))
  229. assertTrue(channel.modesDiscovered)
  230. }
  231. @Test
  232. fun `adds modes when discovered mode event received`() {
  233. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  234. fakeChannelState += channel
  235. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  236. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+ceg", arrayOf("CCC", "EEE"), true))
  237. assertEquals("CCC", channel.modes['c'])
  238. assertEquals("EEE", channel.modes['e'])
  239. assertEquals("", channel.modes['g'])
  240. }
  241. @Test
  242. fun `adjusts complex modes when mode change event received`() {
  243. val channel = ChannelState("#thegibson") { CaseMapping.Rfc }
  244. channel.modes['c'] = "CCC"
  245. channel.modes['e'] = "EEE"
  246. channel.modes['h'] = ""
  247. fakeChannelState += channel
  248. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  249. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "-c+d-eh+fg", arrayOf("CCC", "DDD", "FFF"), true))
  250. assertNull(channel.modes['c'])
  251. assertEquals("DDD", channel.modes['d'])
  252. assertNull(channel.modes['e'])
  253. assertEquals("FFF", channel.modes['f'])
  254. assertEquals("", channel.modes['g'])
  255. assertNull(channel.modes['h'])
  256. }
  257. @Test
  258. fun `handles unprivileged user gaining new mode`() {
  259. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  260. users += ChannelUser("ZeroCool")
  261. fakeChannelState += this
  262. }
  263. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+o", arrayOf("zeroCool")))
  264. assertEquals("o", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  265. }
  266. @Test
  267. fun `handles privileged user gaining lesser mode`() {
  268. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  269. users += ChannelUser("ZeroCool", "o")
  270. fakeChannelState += this
  271. }
  272. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+v", arrayOf("zeroCool")))
  273. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  274. }
  275. @Test
  276. fun `handles privileged user gaining greater mode`() {
  277. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  278. users += ChannelUser("ZeroCool", "v")
  279. fakeChannelState += this
  280. }
  281. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+o", arrayOf("zeroCool")))
  282. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  283. }
  284. @Test
  285. fun `handles user gaining multiple modes`() {
  286. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  287. users += ChannelUser("ZeroCool")
  288. fakeChannelState += this
  289. }
  290. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "+vo", arrayOf("zeroCool", "zeroCool")))
  291. assertEquals("ov", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  292. }
  293. @Test
  294. fun `handles user losing multiple modes`() {
  295. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  296. users += ChannelUser("ZeroCool", "ov")
  297. fakeChannelState += this
  298. }
  299. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "-vo", arrayOf("zeroCool", "zeroCool")))
  300. assertEquals("", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  301. }
  302. @Test
  303. fun `handles mixture of user modes and normal modes`() {
  304. with (ChannelState("#thegibson") { CaseMapping.Rfc }) {
  305. users += ChannelUser("ZeroCool", "v")
  306. fakeChannelState += this
  307. }
  308. fakeServerState.features[ServerFeature.ChannelModes] = arrayOf("ab", "cd", "ef", "gh")
  309. handler.processEvent(ircClient, ModeChanged(EventMetadata(TestConstants.time), "#thegibson", "oa-v+b", arrayOf("zeroCool", "aaa", "zeroCool", "bbb")))
  310. assertEquals("o", fakeChannelState["#thegibson"]?.users?.get("zeroCool")?.modes)
  311. assertEquals("aaa", fakeChannelState["#thegibson"]?.modes?.get('a'))
  312. assertEquals("bbb", fakeChannelState["#thegibson"]?.modes?.get('b'))
  313. }
  314. @Test
  315. fun `updates topic state when it's discovered for the first time`() {
  316. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  317. fakeChannelState += state
  318. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", "Hack the planet!"))
  319. handler.processEvent(ircClient, ChannelTopicMetadataDiscovered(EventMetadata(TestConstants.time), "#thegibson", User("acidBurn"), TestConstants.otherTime))
  320. assertTrue(state.topicDiscovered)
  321. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime), state.topic)
  322. }
  323. @Test
  324. fun `updates topic state when no topic is discovered for the first time`() {
  325. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  326. fakeChannelState += state
  327. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", null))
  328. assertTrue(state.topicDiscovered)
  329. assertEquals(ChannelTopic(), state.topic)
  330. }
  331. @Test
  332. fun `leaves topic state when it's discovered for a second time`() {
  333. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  334. state.topic = ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime)
  335. state.topicDiscovered = true
  336. fakeChannelState += state
  337. handler.processEvent(ircClient, ChannelTopicDiscovered(EventMetadata(TestConstants.time), "#thegibson", "Hack the planet"))
  338. handler.processEvent(ircClient, ChannelTopicMetadataDiscovered(EventMetadata(TestConstants.time), "#thegibson", User("zeroCool"), TestConstants.time))
  339. assertTrue(state.topicDiscovered)
  340. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.otherTime), state.topic)
  341. }
  342. @Test
  343. fun `updates topic state when the topic is changed`() {
  344. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  345. fakeChannelState += state
  346. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#thegibson", "Hack the planet!"))
  347. assertEquals(ChannelTopic("Hack the planet!", User("acidBurn"), TestConstants.time), state.topic)
  348. }
  349. @Test
  350. fun `updates topic state when the topic is unset`() {
  351. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  352. fakeChannelState += state
  353. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#thegibson", null))
  354. assertEquals(ChannelTopic(null, User("acidBurn"), TestConstants.time), state.topic)
  355. }
  356. @Test
  357. fun `ignores topic change when channel doesn't exist`() {
  358. val state = ChannelState("#thegibson") { CaseMapping.Rfc }
  359. fakeChannelState += state
  360. handler.processEvent(ircClient, ChannelTopicChanged(EventMetadata(TestConstants.time), User("acidBurn"), "#dumpsterdiving", "Hack the planet!"))
  361. assertEquals(ChannelTopic(), state.topic)
  362. }
  363. }