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.

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. }