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

org.jboss.aerogear.unifiedpush.message.serviceHolder.AbstractServiceHolder Maven / Gradle / Ivy

There is a newer version: 2.5.0
Show newest version
/**
 * JBoss, Home of Professional Open Source
 * Copyright Red Hat, Inc., and individual contributors.
 *
 * 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.
 */
package org.jboss.aerogear.unifiedpush.message.serviceHolder;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;

import javax.inject.Inject;
import javax.jms.Queue;

import org.jboss.aerogear.unifiedpush.message.util.JmsClient;

/**
 * Holds instances of given service <T> and allows their instantiation, reuse and disposal.
 *
 * Leverages JMS queue to store number of messages corresponding to limit of how many services can be created for given push network.
 *
 * The message is borrowed from this queue when new service is created.The message is returned to this queue when service is destroyed.
 *
 * That ensures that there won't be created more service instances in entire cluster of servers than given limit.
 *
 * @param  the type of the service
 */
public abstract class AbstractServiceHolder {

    private final ConcurrentHashMap>> queueMap = new ConcurrentHashMap>>();

    private final int instanceLimit;
    private final long instanceAcquiringTimeoutInMillis;
    private final long serviceDisposalDelayInMillis;

    @Inject
    private JmsClient jmsClient;

    @Inject
    private ServiceDisposalScheduler serviceDisposalScheduler;

    /**
     * Returns the Queue used as a counter of free services.
     *
     * This queue is populated with number of messages corresponding to limit of how many services can be created for given push network.
     * The message is borrowed from this queue when new service is created.
     * The message is returned to this queue when service is destroyed.
     * That ensures that there won't be created more service instances in entire cluster of servers than given limit.
     *
     * @return the Queue used as a counter of free services.
     */
    public abstract Queue getFreeServiceSlotQueue();

    /**
     * Creates new service instance
     *
     * @param instanceLimit how many instances can be created
     * @param instanceAcquiringTimeoutInMillis what is a timeout before the holder can return null
     * @param serviceDisposalDelayInMillis how long the service instance will be held until it is disposed for inactivity
     */
    public AbstractServiceHolder(int instanceLimit, long instanceAcquiringTimeoutInMillis, long serviceDisposalDelayInMillis) {
        this.instanceLimit = instanceLimit;
        this.instanceAcquiringTimeoutInMillis = instanceAcquiringTimeoutInMillis;
        this.serviceDisposalDelayInMillis = serviceDisposalDelayInMillis;
    }

    public void initialize(final String pushMessageInformationId, final String variantID) {
        for (int i = 0; i < instanceLimit; i++) {
            returnServiceSlotToQueue(pushMessageInformationId, variantID);
        }
    }

    public void destroy(final String pushMessageInformationId, final String variantID) {
        for (int i = 0; i < instanceLimit; i++) {
            if (borrowServiceSlotFromQueue(pushMessageInformationId, variantID) == null) {
                return;
            }
        }
    }

    /**
     * Holder returns a service for given parameters or uses service constructor to instantiate new service.
     *
     * Number of created or queued services is limited up to configured {@link #instanceLimit}.
     *
     * The service blocks until a service is available or configured {@link #instanceAcquiringTimeoutInMillis}.
     *
     * In case the service is not available when times out, holder returns null.
     *
     * @param pushMessageInformationId the push message id
     * @param variantID the variant
     * @param constructor the service constructor
     * @return the service instance; or null in case too much services were created and no services are queued for reuse
     */
    public T dequeueOrCreateNewService(final String pushMessageInformationId, final String variantID, ServiceConstructor constructor) {
        T instance = dequeue(pushMessageInformationId, variantID);
        if (instance != null) {
            return instance;
        }
        // there is no cached instance, try to establish one
        if (borrowServiceSlotFromQueue(pushMessageInformationId, variantID) != null) {
            // we have borrowed a service, we can create new instance
            return constructor.construct();
        }
        return null;
    }

    /**
     * Dequeues the service instance if there is one available, otherwise returns null
     * @param pushMessageInformationId the push message id
     * @param variantID the variant
     * @return the service instance or null if no instance is queued
     */
    public T dequeue(final String pushMessageInformationId, final String variantID) {
        ConcurrentLinkedQueue> concurrentLinkedQueue = getCache(pushMessageInformationId, variantID);
        DisposableReference serviceHolder;
        // poll queue for new instance
        while ((serviceHolder = concurrentLinkedQueue.poll()) != null) {
            T serviceInstance = serviceHolder.get();
            // holder may hold expired instance
            if (serviceInstance != null) {
                return serviceInstance;
            }
        }
        return null;
    }

    /**
     * Allows to queue used and freed up service into cache so that can be reused by another consumer.
     *
     * @param pushMessageInformationId the push message
     * @param variantID the variant
     * @param service the used and freed up service
     * @param destroyer the instance of {@link ServiceDestroyer} used to destroy service instance
     */
    public void queueFreedUpService(final String pushMessageInformationId, final String variantID, final T service, final ServiceDestroyer destroyer) {
        ServiceDestroyer destroyAndReturnServiceSlot = new ServiceDestroyer() {
            @Override
            public void destroy(T instance) {
                destroyer.destroy(instance);
                returnServiceSlotToQueue(pushMessageInformationId, variantID);
            };
        };
        DisposableReference disposableReference = new DisposableReference(service, destroyAndReturnServiceSlot);
        serviceDisposalScheduler.scheduleForDisposal(disposableReference, serviceDisposalDelayInMillis);
        getCache(pushMessageInformationId, variantID).add(disposableReference);
    }

    /**
     * Allows to free up a counter of created services and thus allowing waiting consumers to create new services within the limits.
     *
     * Freed up service is a service that died, disconnected or similar and can no longer be used.
     *
     * @param pushMessageInformationId the push message
     * @param variantID the variant
     */
    public void freeUpSlot(final String pushMessageInformationId, final String variantID) {
        returnServiceSlotToQueue(pushMessageInformationId, variantID);
    }

    protected Object borrowServiceSlotFromQueue(String pushMessageInformationId, String variantID) {
        return jmsClient.receive().withSelector("variantID = '%s'", variantID).withTimeout(instanceAcquiringTimeoutInMillis).from(getFreeServiceSlotQueue());
    }

    protected void returnServiceSlotToQueue(String pushMessageInformationId, String variantID) {
        jmsClient.send(pushMessageInformationId + ":" + variantID).withProperty("variantID", variantID).to(getFreeServiceSlotQueue());
    }

    private ConcurrentLinkedQueue> getCache(String pushMessageInformationId, String variantID) {
        return getOrCreateQueue(new Key(pushMessageInformationId, variantID));
    }

    private ConcurrentLinkedQueue> getOrCreateQueue(Key key) {
        ConcurrentLinkedQueue> queue = queueMap.get(key);
        if (queue == null) {
            queue = queueMap.putIfAbsent(key, new ConcurrentLinkedQueue>());
            queue = queueMap.get(key);
        }
        return queue;
    }

    /**
     * The key that is used to store a queue instance in the map.
     */
    private static class Key {

        private String pushMessageInformationId;
        private String variantId;

        Key (String pushMessageInformationId, String variantID) {
            if (pushMessageInformationId == null) {
                throw new NullPointerException("pushMessageInformationId");
            }
            if (variantID == null) {
                throw new NullPointerException("variant or its variantID cant be null");
            }
            this.pushMessageInformationId = pushMessageInformationId;
            this.variantId = variantID;
        }

        @Override
        public int hashCode() {
            final int prime = 31;
            int result = 1;
            result = prime * result + ((pushMessageInformationId == null) ? 0 : pushMessageInformationId.hashCode());
            result = prime * result + ((variantId == null) ? 0 : variantId.hashCode());
            return result;
        }

        @Override
        public boolean equals(Object obj) {
            if (this == obj)
                return true;
            if (obj == null)
                return false;
            if (getClass() != obj.getClass())
                return false;
            Key other = (Key) obj;
            if (pushMessageInformationId == null) {
                if (other.pushMessageInformationId != null)
                    return false;
            } else if (!pushMessageInformationId.equals(other.pushMessageInformationId))
                return false;
            if (variantId == null) {
                if (other.variantId != null)
                    return false;
            } else if (!variantId.equals(other.variantId))
                return false;
            return true;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy