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

se.laz.casual.jca.work.StartInboundServerWork Maven / Gradle / Ivy

There is a newer version: 3.3.0
Show newest version
/*
 * Copyright (c) 2021 - 2024, The casual project. All rights reserved.
 *
 * This software is licensed under the MIT license, https://opensource.org/licenses/MIT
 */

package se.laz.casual.jca.work;

import jakarta.resource.spi.work.Work;
import se.laz.casual.config.ConfigurationOptions;
import se.laz.casual.config.ConfigurationService;
import se.laz.casual.jca.InboundStartupException;
import se.laz.casual.jca.inbound.handler.service.ServiceHandler;
import se.laz.casual.jca.inbound.handler.service.ServiceHandlerFactory;
import se.laz.casual.jca.inbound.handler.service.ServiceHandlerNotFoundException;

import java.util.HashSet;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Supplier;
import java.util.logging.Logger;
import java.util.stream.Collectors;

/**
 * Work instance for delaying start of inbound server until all startup services have been registered
 */
public final class StartInboundServerWork implements Work
{
    private static Logger log = Logger.getLogger( StartInboundServerWork.class.getName());
    private final List startupServices;
    private final Consumer consumer;
    private final Supplier supplier;
    private final Supplier logMessage;
    private long delay;

    private StartInboundServerWork( List startupServices, Supplier logMessage, Consumer consumer, Supplier supplier, long delay)
    {
        this.startupServices = startupServices;
        this.consumer = consumer;
        this.supplier = supplier;
        this.logMessage = logMessage;
        this.delay = delay;
    }

    public static  Work of(List startupServices, Supplier logMessage, Consumer consumer, Supplier supplier )
    {
        return of(startupServices, logMessage, consumer, supplier, 0);
    }

    public static  Work of(List startupServices, Supplier logMessage, Consumer consumer, Supplier supplier, long delay )
    {
        Objects.requireNonNull(startupServices, "Startup Services is null.");
        Objects.requireNonNull(consumer, "Consumer is null.");
        Objects.requireNonNull(supplier, "supplier is null");
        return new StartInboundServerWork<>(startupServices, logMessage, consumer, supplier, delay);
    }

    @Override
    public void release()
    {
        /**
         * We can not speed up the completion
         */
    }

    @Override
    public void run()
    {
        waitForInboundStartupServices();
        maybeDelay();
        startInboundServer();
    }

    private void waitForInboundStartupServices()
    {
        logInitialStartupServices(startupServices);
        Set remaining = checkRemainingServices( new HashSet<>( this.startupServices ) );
        while(!remaining.isEmpty())
        {
            int previousNumberOfServicesRemaining = remaining.size();
            remaining = checkRemainingServices( remaining );
            if(!remaining.isEmpty() && remaining.size() < previousNumberOfServicesRemaining)
            {
                logWaitingForStartupServices(remaining);
            }
            try
            {
                Thread.sleep( 1000 );
            }
            catch( InterruptedException e )
            {
                Thread.currentThread().interrupt();
                throw new InboundStartupException( "Interrupted waiting for inbound startup services registration.", e );
            }
        }
        log.info(() -> "All startup services registered.");
    }

    private void logInitialStartupServices(List startupServices)
    {
        log.info(() -> "Inbound startup mode: " + ConfigurationService.getConfiguration( ConfigurationOptions.CASUAL_INBOUND_STARTUP_MODE ) );
        log.info(() -> "Waiting for " + startupServices.size() + " services to be registered before inbound starts.");
        log.info(() -> "Initial services list: " + startupServices.stream()
                                                                  .collect(Collectors.joining()));
    }

    private void logWaitingForStartupServices(Set remaining)
    {
        log.info(() -> "Waiting for registration of the following services: " + remaining.stream()
                                                                                         .collect(Collectors.joining()));
    }

    private Set checkRemainingServices( Set remaining )
    {
        Set notFound = new HashSet<>( remaining );
        for( String find: remaining )
        {
            try
            {
                ServiceHandler handler = ServiceHandlerFactory.getHandler( find );
                if( handler.isServiceAvailable( find ) )
                {
                    notFound.remove( find );
                    log.info( () -> "Startup service registered: " + find );
                }

            }
            catch( ServiceHandlerNotFoundException e )
            {
                //Service not registered.
            }
        }
        return notFound;
    }

    private void maybeDelay()
    {
        // Note:
        // This, optional, delay of inbound is due to an issue on wls where once in a blue moon - the MessageEndpointFactory is in fact not ready
        // and there are incoming domain discoveries ( during restart)
        // That will then fail and continue to fail forever, even after endpointActivation for the resource adapter has completed - it should heal
        // This is a horrible "work around" for that specific problem - the user decides how long inbound startup has to be delayed
        // Never seen on wildfly
        if(delay <= 0L)
        {
            log.info(() -> "no inbound startup delay");
            return;
        }
        Delayer.delay(delay);
        log.info(() -> "inbound startup, delay of " + delay + " seconds - done");
    }

    private void startInboundServer()
    {
        consumer.accept(supplier.get());
        log.info(logMessage::get);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy