選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

MessageHandler.kt 2.3KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162
  1. package com.dmdirc.ktirc.io
  2. import com.dmdirc.ktirc.IrcClient
  3. import com.dmdirc.ktirc.events.IrcEvent
  4. import com.dmdirc.ktirc.events.handlers.EventHandler
  5. import com.dmdirc.ktirc.events.mutators.EventMutator
  6. import com.dmdirc.ktirc.messages.MessageProcessor
  7. import com.dmdirc.ktirc.model.IrcMessage
  8. import com.dmdirc.ktirc.util.logger
  9. import kotlinx.coroutines.channels.ReceiveChannel
  10. internal interface MessageEmitter {
  11. fun handleEvent(ircClient: IrcClient, ircEvent: IrcEvent, processOnly: Boolean = false) = handleEvents(ircClient, listOf(ircEvent), processOnly)
  12. fun handleEvents(ircClient: IrcClient, ircEvents: List<IrcEvent>, processOnly: Boolean = false)
  13. }
  14. internal class MessageHandler(
  15. private val processors: List<MessageProcessor>,
  16. private val mutators: List<EventMutator>,
  17. private val handlers: List<EventHandler>) : MessageEmitter {
  18. private val log by logger()
  19. private val emitters = mutableListOf<(IrcEvent) -> Unit>()
  20. suspend fun processMessages(ircClient: IrcClient, messages: ReceiveChannel<IrcMessage>) {
  21. for (message in messages) {
  22. handleEvents(ircClient, message.toEvents())
  23. }
  24. }
  25. override fun handleEvents(ircClient: IrcClient, ircEvents: List<IrcEvent>, processOnly: Boolean) {
  26. val events = if (processOnly) ircEvents else ircEvents.mutate(ircClient)
  27. events.forEach { event ->
  28. event.process(ircClient)
  29. if (!processOnly) {
  30. log.fine { "Dispatching event of type ${event::class}" }
  31. emitters.forEach { it(event) }
  32. }
  33. }
  34. }
  35. fun addEmitter(emitter: (IrcEvent) -> Unit) {
  36. emitters.add(emitter)
  37. }
  38. private fun IrcEvent.process(ircClient: IrcClient) = handlers.forEach { it.processEvent(ircClient, this) }
  39. private fun List<IrcEvent>.mutate(ircClient: IrcClient) = mutators.fold(this) { events, mutator ->
  40. events.flatMap { mutator.mutateEvent(ircClient, this@MessageHandler, it) }
  41. }
  42. private fun IrcMessage.toEvents() = this.getProcessor()?.process(this) ?: emptyList()
  43. private fun IrcMessage.getProcessor() = processors.firstOrNull { it.commands.contains(command) } ?: run {
  44. log.warning { "No processor found for $command" }
  45. null
  46. }
  47. }