All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.elasticmq.persistence.sql.SqlQueuePersistenceActor.scala Maven / Gradle / Ivy

package org.elasticmq.persistence.sql

import org.apache.pekko.actor.{Actor, ActorRef}
import org.apache.pekko.util.Timeout
import org.elasticmq.ElasticMQError
import org.elasticmq.actor.queue._
import org.elasticmq.actor.reply._
import org.elasticmq.msg.{CreateQueue, RestoreMessages}
import org.elasticmq.persistence.CreateQueueMetadata
import org.elasticmq.util.Logging

import scala.collection.mutable
import scala.concurrent.{ExecutionContext, Future}
import scala.concurrent.duration.DurationInt
import scala.util.{Failure, Success}

case class GetAllMessages(queueName: String) extends Replyable[List[InternalMessage]]

class SqlQueuePersistenceActor(
    messagePersistenceConfig: SqlQueuePersistenceConfig,
    baseQueues: List[CreateQueueMetadata]
) extends Actor
    with Logging {

  private val db = new DB(messagePersistenceConfig)
  private val queueRepo = new QueueRepository(db)
  private val repos = mutable.HashMap[String, MessageRepository]()

  implicit val timeout: Timeout = Timeout(5.seconds)
  implicit val ec: ExecutionContext = context.dispatcher

  def receive: Receive = {
    case QueueEvent.QueueCreated(queueData) =>
      logger.debug(s"Storing queue data: $queueData")

      if (repos.contains(queueData.name)) {
        val _: Int = queueRepo.update(CreateQueueMetadata.from(queueData))
      } else {
        queueRepo.add(CreateQueueMetadata.from(queueData))
        repos.put(queueData.name, new MessageRepository(queueData.name, db))
        ()
      }

    case QueueEvent.QueueDeleted(queueName) =>
      logger.debug(s"Removing queue data for queue $queueName")
      val _: Int = queueRepo.remove(queueName)
      repos.remove(queueName).foreach(_.drop())

    case QueueEvent.QueueMetadataUpdated(queueData) =>
      logger.whenDebugEnabled {
        logger.debug(s"Updating queue: $queueData")
      }
      queueRepo.update(CreateQueueMetadata.from(queueData))
      ()

    case QueueEvent.MessageAdded(queueName, message) =>
      logger.whenDebugEnabled {
        logger.debug(s"Adding new message: $message")
      }
      repos.get(queueName).foreach(_.add(message))
      sender() ! OperationSuccessful

    case QueueEvent.MessageUpdated(queueName, message) =>
      logger.whenDebugEnabled {
        logger.debug(s"Updating message: $message")
      }
      repos.get(queueName).foreach(_.update(message))
      sender() ! OperationSuccessful

    case QueueEvent.MessageRemoved(queueName, messageId) =>
      logger.whenDebugEnabled {
        logger.debug(s"Removing message with id $messageId")
      }
      repos.get(queueName).foreach(_.remove(messageId))
      sender() ! OperationSuccessful

    case QueueEvent.Restore(queueManagerActor: ActorRef) =>
      val recip = sender()
      createQueues(queueManagerActor).onComplete {
        case Success(result)    => recip ! result
        case Failure(exception) => logger.error("Failed to restore persisted queues", exception)
      }

    case GetAllMessages(queueName) =>
      repos.get(queueName).foreach(repo => sender() ! repo.findAll())
  }

  private def createQueues(
      queueManagerActor: ActorRef
  ): Future[Either[List[ElasticMQError], OperationStatus]] = {
    val persistedQueues = queueRepo.findAll()
    val allQueues = CreateQueueMetadata.mergePersistedAndBaseQueues(persistedQueues, baseQueues)

    allQueues.foreach(queue => repos.put(queue.name, new MessageRepository(queue.name, db)))

    val restoreResult: List[Future[Either[ElasticMQError, OperationStatus]]] = allQueues.map { cq =>
      restoreQueue(queueManagerActor, cq)
        .flatMap {
          case Left(errors)      => Future.successful(Left(errors))
          case Right(queueActor) => restoreMessages(cq.name, queueActor).map(_ => Right(OperationSuccessful))
        }
    }

    Future
      .sequence(restoreResult)
      .map(results => {
        val errors = results.flatMap(_.swap.toOption)
        if (errors.nonEmpty) Left(errors) else Right(OperationSuccessful)
      })
  }

  private def restoreQueue(
      queueManagerActor: ActorRef,
      cq: CreateQueueMetadata
  ): Future[Either[ElasticMQError, ActorRef]] = {
    queueManagerActor ? CreateQueue(cq.toCreateQueueData)
  }

  private def restoreMessages(queueName: String, queueActor: ActorRef): Future[Unit] = {
    queueActor ? RestoreMessages(repos(queueName).findAll())
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy