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

org.asteriskjava.live.internal.AsteriskQueueImpl Maven / Gradle / Ivy

There is a newer version: 3.39.0
Show newest version
/*
 *  Copyright 2004-2006 Stefan Reuter
 *
 *  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.asteriskjava.live.internal;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Timer;
import java.util.TimerTask;

import org.asteriskjava.live.AsteriskQueue;
import org.asteriskjava.live.AsteriskQueueEntry;
import org.asteriskjava.live.AsteriskQueueListener;
import org.asteriskjava.live.AsteriskQueueMember;
import org.asteriskjava.util.AstUtil;
import org.asteriskjava.util.Log;
import org.asteriskjava.util.LogFactory;

/**
 * Default implementation of the AsteriskQueue interface.
 *
 * @author srt
 * @version $Id$
 */
class AsteriskQueueImpl extends AbstractLiveObject implements AsteriskQueue
{
    /**
     * TimerTask that monitors exceeding service levels.
     *
     * @author Patrick Breucking
     */
    private class ServiceLevelTimerTask extends TimerTask
    {
        private final AsteriskQueueEntry entry;

        ServiceLevelTimerTask(AsteriskQueueEntry entry)
        {
            this.entry = entry;
        }

        @Override
        public void run()
        {
            fireServiceLevelExceeded(entry);
        }
    }

    private final Log logger = LogFactory.getLog(this.getClass());
    private final String name;
    private Integer max;
    private String strategy;
    private Integer serviceLevel;

    /** 101118 Octavio Luna Agregado para Ver cambios en Vivo **/
    private Integer calls;
    private Integer holdTime;
    private Integer talkTime;
    private Integer completed;
    private Integer abandoned;
    private Double serviceLevelPerf;
    /****/

    private Integer weight;
    private final List entries;
    private final Timer timer;
    private final Map members;
    private final List listeners;
    private final Map serviceLevelTimerTasks;

    AsteriskQueueImpl(AsteriskServerImpl server, String name, Integer max,
                      String strategy, Integer serviceLevel, Integer weight,
                      Integer calls, Integer holdTime, Integer talkTime,
                      Integer completed, Integer abandoned, Double serviceLevelPerf)
    {
        super(server);
        this.name = name;
        this.max = max;
        this.strategy = strategy;
        this.serviceLevel = serviceLevel;
        this.weight = weight;
        entries = new ArrayList<>(25);
        listeners = new ArrayList<>();
        members = new HashMap<>();
        timer = new Timer("ServiceLevelTimer-" + name, true);
        serviceLevelTimerTasks = new HashMap<>();
        this.calls = calls;
        this.holdTime = holdTime;
        this.talkTime = talkTime;
        this.completed = completed;
        this.abandoned = abandoned;
        this.serviceLevelPerf = serviceLevelPerf;

        stampLastUpdate();
    }

    void cancelServiceLevelTimer()
    {
        timer.cancel();
    }

    public String getName()
    {
        return name;
    }

    public Integer getMax()
    {
        return max;
    }

    public String getStrategy()
    {
        return strategy;
    }

    /**
     *
     * @param max
     * @return true if value updated, false otherwise
     */
    boolean setMax(Integer max)
    {
    	if(!AstUtil.isEqual(this.max, max)){
    		this.max = max;
			stampLastUpdate();
			return true;
		}
    	return false;
    }

    public Integer getServiceLevel()
    {
        return serviceLevel;
    }

    /**
     *
     * @param serviceLevel
     * @return
     */
    boolean setServiceLevel(Integer serviceLevel)
    {
    	if(!AstUtil.isEqual(this.serviceLevel, serviceLevel)){
    		this.serviceLevel = serviceLevel;
			stampLastUpdate();
			return true;
		}
    	return false;

    }

    @Override
	public Integer getCalls() {
		return calls;
	}

    /**
     *
     * @param calls
     * @return  true if value updated, false otherwise
     */
    boolean setCalls(Integer calls) {
    	if(!AstUtil.isEqual(this.calls, calls)){
    		this.calls = calls;
			stampLastUpdate();
			return true;
		}
    	return false;
	}

    @Override
    public Integer getWaiting() {
    	return calls;
    }

    @Override
	public Integer getHoldTime() {
		return holdTime;
	}

    /**
     *
     * @param holdTime
     * @return true if value updated, false otherwise
     */
    boolean setHoldTime(Integer holdTime) {
    	if(!AstUtil.isEqual(this.holdTime, holdTime)){
    		this.holdTime = holdTime;
			stampLastUpdate();
			return true;
		}
    	return false;
	}

    @Override
	public Integer getTalkTime() {
		return talkTime;
	}

    /**
     *
     * @param talkTime
     * @return true if value updated, false otherwise
     */
    boolean setTalkTime(Integer talkTime) {
    	if(!AstUtil.isEqual(this.talkTime, talkTime)){
    		this.talkTime = talkTime;
			stampLastUpdate();
			return true;
		}
    	return false;
	}

    @Override
	public Integer getCompleted() {
		return completed;
	}

    /**
     *
     * @param completed
     * @return  true if value updated, false otherwise
     */
    boolean setCompleted(Integer completed) {
    	if(!AstUtil.isEqual(this.completed, completed)){
    		this.completed = completed;
			stampLastUpdate();
			return true;
		}
    	return false;
	}

    @Override
	public Integer getAbandoned() {
		return abandoned;
	}

    /**
     *
     * @param abandoned
     * @return true if value updated, false otherwise
     */
	boolean setAbandoned(Integer abandoned) {
		if(!AstUtil.isEqual(this.abandoned, abandoned)){
			this.abandoned = abandoned;
			stampLastUpdate();
			return true;
		}

		return false;
	}

	@Override
	public Double getServiceLevelPerf() {
		return serviceLevelPerf;
	}

	/**
	 *
	 * @param serviceLevelPerf
	 * @return true if value updated, false otherwise
	 */
	boolean setServiceLevelPerf(Double serviceLevelPerf) {
		if(!AstUtil.isEqual(this.serviceLevelPerf, serviceLevelPerf)){
			this.serviceLevelPerf = serviceLevelPerf;
			stampLastUpdate();
			return true;
		}
		return false;
	}

    public Integer getWeight()
    {
        return weight;
    }

    /**
     *
     * @param weight
     * @return true if value updated, false otherwise
     */
    boolean setWeight(Integer weight)
    {
    	if(!AstUtil.isEqual(this.weight, weight)){
    		this.weight = weight;
			stampLastUpdate();
			return true;
		}
    	return false;
    }

    public List getEntries()
    {
        synchronized (entries)
        {
            return new ArrayList(entries);
        }
    }

    /**
     * Shifts the position of the queue entries if needed
     * (and fire PCE on queue entries if appropriate).
     */
    private void shift()
    {
        int currentPos = 1; // Asterisk starts at 1

        synchronized (entries)
        {
            for (AsteriskQueueEntryImpl qe : entries)
            {
                // Only set (and fire PCE on qe) if necessary
                if (qe.getPosition() != currentPos)
                {
                    qe.setPosition(currentPos);
                }
                currentPos++;
            }
        }
    }

    /**
     * Creates a new AsteriskQueueEntry, adds it to this queue.

* Fires: *

    *
  • PCE on channel
  • *
  • NewEntry on this queue
  • *
  • PCE on other queue entries if shifted (never happens)
  • *
  • NewQueueEntry on server
  • *
* * @param channel the channel that joined the queue * @param reportedPosition the position as given by Asterisk (currently not used) * @param dateReceived the date the hannel joined the queue */ void createNewEntry(AsteriskChannelImpl channel, int reportedPosition, Date dateReceived) { AsteriskQueueEntryImpl qe = new AsteriskQueueEntryImpl(server, this, channel, reportedPosition, dateReceived); long delay = serviceLevel * 1000L; if (delay > 0) { ServiceLevelTimerTask timerTask = new ServiceLevelTimerTask(qe); timer.schedule(timerTask, delay); synchronized (serviceLevelTimerTasks) { serviceLevelTimerTasks.put(qe, timerTask); } } synchronized (entries) { entries.add(qe); // at the end of the list // Keep the lock ! // This will fire PCE on the newly created queue entry // but hopefully this one has no listeners yet shift(); } // Set the channel property ony here as queue entries and channels // maintain a reciprocal reference. // That way property change on channel and new entry event on queue will be // lanched when BOTH channel and queue are correctly set. channel.setQueueEntry(qe); fireNewEntry(qe); server.fireNewQueueEntry(qe); } /** * Removes the given queue entry from the queue.

* Fires if needed: *

    *
  • PCE on channel
  • *
  • EntryLeave on this queue
  • *
  • PCE on other queue entries if shifted
  • *
* * @param entry an existing entry object. * @param dateReceived the remove event was received. */ void removeEntry(AsteriskQueueEntryImpl entry, Date dateReceived) { synchronized (serviceLevelTimerTasks) { if (serviceLevelTimerTasks.containsKey(entry)) { ServiceLevelTimerTask timerTask = serviceLevelTimerTasks.get(entry); timerTask.cancel(); serviceLevelTimerTasks.remove(entry); } } boolean changed; synchronized (entries) { changed = entries.remove(entry); if (changed) { // Keep the lock ! shift(); } } // Fire outside lock if (changed) { entry.getChannel().setQueueEntry(null); entry.left(dateReceived); fireEntryLeave(entry); } } @Override public String toString() { final StringBuilder sb; sb = new StringBuilder("AsteriskQueue["); sb.append("name='").append(getName()).append("',"); sb.append("max='").append(getMax()).append("',"); sb.append("strategy='").append(getStrategy()).append("',"); sb.append("serviceLevel='").append(getServiceLevel()).append("',"); sb.append("weight='").append(getWeight()).append("',"); sb.append("calls='").append(getCalls()).append("',"); sb.append("holdTime='").append(getHoldTime()).append("',"); sb.append("talkTime='").append(getTalkTime()).append("',"); sb.append("completed='").append(getCompleted()).append("',"); sb.append("abandoned='").append(getAbandoned()).append("',"); sb.append("serviceLevelPerf='").append(getServiceLevelPerf()).append("',"); synchronized (entries) { sb.append("entries='").append(entries.toString()).append("',"); } synchronized (members) { sb.append("members='").append(members.toString()).append("',"); } sb.append("systemHashcode=").append(System.identityHashCode(this)); sb.append("]"); return sb.toString(); } public void addAsteriskQueueListener(AsteriskQueueListener listener) { synchronized (listeners) { listeners.add(listener); } } public void removeAsteriskQueueListener(AsteriskQueueListener listener) { synchronized (listeners) { listeners.remove(listener); } } /** * Notifies all registered listener that an entry joins the queue. * * @param entry that joins the queue */ void fireNewEntry(AsteriskQueueEntryImpl entry) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onNewEntry(entry); } catch (Exception e) { logger.warn("Exception in onNewEntry()", e); } } } } /** * Notifies all registered listener that an entry leaves the queue. * * @param entry that leaves the queue. */ void fireEntryLeave(AsteriskQueueEntryImpl entry) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onEntryLeave(entry); } catch (Exception e) { logger.warn("Exception in onEntryLeave()", e); } } } } /** * Notifies all registered listener that a member has been added to the queue. * * @param member added to the queue */ void fireMemberAdded(AsteriskQueueMemberImpl member) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onMemberAdded(member); } catch (Exception e) { logger.warn("Exception in onMemberAdded()", e); } } } } /** * Notifies all registered listener that a member has been removed from the queue. * * @param member that has been removed. */ void fireMemberRemoved(AsteriskQueueMemberImpl member) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onMemberRemoved(member); } catch (Exception e) { logger.warn("Exception in onMemberRemoved()", e); } } } } /** * Returns a collection of members of this queue. * * @see org.asteriskjava.live.AsteriskQueue#getMembers() */ public Collection getMembers() { List listOfMembers = new ArrayList<>(members.size()); synchronized (members) { for (AsteriskQueueMemberImpl asteriskQueueMember : members.values()) { listOfMembers.add(asteriskQueueMember); } } return listOfMembers; } /** * Returns a member by its location. * * @param location ot the member * @return the member by its location. */ AsteriskQueueMemberImpl getMember(String location) { synchronized (members) { if (members.containsKey(location)) { return members.get(location); } } return null; } /** * Add a new member to this queue. * * @param member to add */ void addMember(AsteriskQueueMemberImpl member) { synchronized (members) { // Check if member already exists if (members.containsValue(member)) { return; } // If not, add the new member. logger.info("Adding new member to the queue " + getName() + ": " + member.toString()); members.put(member.getLocation(), member); } fireMemberAdded(member); } /** * Retrieves a member by its location. * * @param location of the member * @return the requested member. */ AsteriskQueueMemberImpl getMemberByLocation(String location) { AsteriskQueueMemberImpl member; synchronized (members) { member = members.get(location); } if (member == null) { logger.error("Requested member at location " + location + " not found!"); } return member; } /** * Notifies all registered listener that a queue member changes its state. * * @param member the changed member. */ void fireMemberStateChanged(AsteriskQueueMemberImpl member) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onMemberStateChange(member); } catch (Exception e) { logger.warn("Exception in onMemberStateChange()", e); } } } } /** * Gets an entry of the queue by its channel name. * * @param channelName The entry's channel name. * @return the queue entry if found, null otherwise. */ AsteriskQueueEntryImpl getEntry(String channelName) { synchronized (entries) { for (AsteriskQueueEntryImpl entry : entries) { if (entry.getChannel().getName().equals(channelName)) { return entry; } } } return null; } /** * Removes a member from this queue. * * @param member the member to remove. */ public void removeMember(AsteriskQueueMemberImpl member) { synchronized (members) { // Check if member exists if (!members.containsValue(member)) { return; } // If so, remove the member. logger.info("Remove member from the queue " + getName() + ": " + member.toString()); members.remove(member.getLocation()); } fireMemberRemoved(member); } void fireServiceLevelExceeded(AsteriskQueueEntry entry) { synchronized (listeners) { for (AsteriskQueueListener listener : listeners) { try { listener.onEntryServiceLevelExceeded(entry); } catch (Exception e) { logger.warn("Exception in fireServiceLevelExceeded()", e); } } } } /** * Gets an entry by its (estimated) position in the queue. * * @param position the position, starting at 1. * @return the queue entry if exiting at this position, null otherwise. */ AsteriskQueueEntryImpl getEntry(int position) { // positions in asterisk start at 1, but list starts at 0 position--; AsteriskQueueEntryImpl foundEntry = null; synchronized (entries) { try { foundEntry = entries.get(position); } catch (IndexOutOfBoundsException e) { // For consistency with the above method, // swallow. We might indeed request the 1st one from time to time } // NOPMD } return foundEntry; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy