RoundRobinAsyncMessageDelivery.java

/*
 * *************************************************************************************************************************************************************
 *
 * TheseFoolishThings: Miscellaneous utilities
 * http://tidalwave.it/projects/thesefoolishthings
 *
 * Copyright (C) 2009 - 2025 by Tidalwave s.a.s. (http://tidalwave.it)
 *
 * *************************************************************************************************************************************************************
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
 * CONDITIONS OF ANY KIND, either express or implied.  See the License for the specific language governing permissions and limitations under the License.
 *
 * *************************************************************************************************************************************************************
 *
 * git clone https://bitbucket.org/tidalwave/thesefoolishthings-src
 * git clone https://github.com/tidalwave-it/thesefoolishthings-src
 *
 * *************************************************************************************************************************************************************
 */
package it.tidalwave.messagebus.spi;

import javax.annotation.Nonnull;
import it.tidalwave.messagebus.spi.MultiQueue.TopicAndMessage;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
import lombok.extern.slf4j.Slf4j;

/***************************************************************************************************************************************************************
 *
 * An implementation of {@link MessageDelivery} that dispatches messages in a round-robin fashion, topic by topic.
 * Each delivery is performed in a separated thread.
 * 
 * @author  Fabrizio Giudici
 * @since   2.2
 *
 **************************************************************************************************************************************************************/
@Slf4j @ToString(of = "workers")
public class RoundRobinAsyncMessageDelivery implements MessageDelivery
  {
    @Nonnull
    private SimpleMessageBus messageBusSupport;
    
    @Getter @Setter
    private int workers = 10;
    
    private final MultiQueue multiQueue = new MultiQueue();
    
    /***********************************************************************************************************************************************************
     **********************************************************************************************************************************************************/
    private final Runnable dispatcher = new Runnable()
      {
        @Override
        public void run() 
          {
            for (;;)
              {
                try 
                  {
                    dispatchMessage(multiQueue.remove());
                  } 
                catch (InterruptedException e) 
                  {
                    break;
                  }
              }
          }
        
        private <T> void dispatchMessage (@Nonnull final TopicAndMessage<T> tam)
          {
            messageBusSupport.dispatchMessage(tam.getTopic(), tam.getMessage());
          }
      };

    /***********************************************************************************************************************************************************
     * {@inheritDoc}
     **********************************************************************************************************************************************************/
    @Override
    public void initialize (@Nonnull final SimpleMessageBus messageBusSupport)
      {
        this.messageBusSupport = messageBusSupport;
        final var executor = this.messageBusSupport.getExecutor();
        
        for (var i = 0; i < workers; i++)
          {
            executor.execute(dispatcher);
          }
      }
    
    /***********************************************************************************************************************************************************
     * {@inheritDoc}
     **********************************************************************************************************************************************************/
    @Override
    public <T> void deliverMessage (@Nonnull final Class<T> topic, @Nonnull final T message)
      {
        multiQueue.add(topic, message);
      }
  }