Package com.sun.enterprise.management.support

Source Code of com.sun.enterprise.management.support.NotificationServiceImpl$RegistrationListener

/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 1997-2007 Sun Microsystems, Inc. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License").  You
* may not use this file except in compliance with the License. You can obtain
* a copy of the License at https://glassfish.dev.java.net/public/CDDL+GPL.html
* or glassfish/bootstrap/legal/LICENSE.txt.  See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at glassfish/bootstrap/legal/LICENSE.txt.
* Sun designates this particular file as subject to the "Classpath" exception
* as provided by Sun in the GPL Version 2 section of the License file that
* accompanied this code.  If applicable, add the following below the License
* Header, with the fields enclosed by brackets [] replaced by your own
* identifying information: "Portions Copyrighted [year]
* [name of copyright owner]"
*
* Contributor(s):
*
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license."  If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above.  However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package com.sun.enterprise.management.support;

import java.util.Set;
import java.util.HashSet;
import java.util.List;
import java.util.LinkedList;
import java.util.Map;
import java.util.HashMap;
import java.util.Collections;
import java.io.Serializable;

import javax.management.MBeanServer;
import javax.management.ObjectName;
import javax.management.Notification;
import javax.management.MBeanServerNotification;
import javax.management.NotificationListener;
import javax.management.NotificationFilter;
import javax.management.NotificationFilterSupport;
import javax.management.ListenerNotFoundException;
import javax.management.InstanceNotFoundException;
import javax.management.NotificationEmitter;
import javax.management.MBeanNotificationInfo;


import com.sun.appserv.management.base.NotificationService;
import com.sun.appserv.management.base.XTypes;
import com.sun.appserv.management.base.AMX;

import com.sun.appserv.management.util.misc.GSetUtil;
import com.sun.appserv.management.util.misc.OverflowHandler;
import com.sun.appserv.management.util.jmx.JMXUtil;
import com.sun.appserv.management.util.jmx.NotificationBuilder;



/**
*/
public final class NotificationServiceImpl extends AMXImplBase
  implements NotificationListener, NotificationEmitter
{
  private Map<ObjectName,NotificationFilter>  mListenees;
  private final Object          mUserData;
 
  private final Map<ObjectName,NotificationFilter> mIncludePatterns;
  private final Map<ObjectName,NotificationFilter> mExcludePatterns;
 
  private final Map<Object,NotificationBuffer>    mBuffers;
 
    public
  NotificationServiceImpl(
    final Object  userData,
    final int    bufferSize  )
  {
    if ( userData == null || !(userData instanceof Serializable) )
    {
      throw new IllegalArgumentException();
    }
   
    mUserData  = userData;

    mListenees    = Collections.synchronizedMap( new HashMap<ObjectName,NotificationFilter>() );
   
    mBuffers    = Collections.synchronizedMap( new HashMap<Object,NotificationBuffer>() );
   
    mIncludePatterns  = new HashMap<ObjectName,NotificationFilter>();
    mExcludePatterns  = new HashMap<ObjectName,NotificationFilter>();
  }
 
    public final Object
  getUserData()
  {
    return( mUserData );
  }
 
  private final String[]  NOTIF_TYPES  = new String[]
  {
    NotificationService.BUFFER_OVERFLOW_NOTIFICATION_TYPE,
  };
 
    public MBeanNotificationInfo[]
  getNotificationInfo()
  {
    final MBeanNotificationInfo  info  = new MBeanNotificationInfo(
      NOTIF_TYPES,
      Notification.class.getName(),
      "" );
    final MBeanNotificationInfo[]  selfInfos  = new MBeanNotificationInfo[] { info };
   
    return( JMXUtil.mergeMBeanNotificationInfos( super.getNotificationInfo(), selfInfos ) );
  }

  /**
   */
    protected void
  issueBufferOverflowNotification( final Notification  oldNotif )
  {
    if ( shouldEmitNotifications() )
    {
      sendNotification( NotificationService.BUFFER_OVERFLOW_NOTIFICATION_TYPE,
          NotificationService.OVERFLOWED_NOTIFICATION_KEY,
          oldNotif );
    }
  }

    public void
  handleNotification(
    final Notification  notif,
    final Object    handback)
  {
    synchronized( mBuffers )
    {
      for( final Object id : mBuffers.keySet() )
      {
        getBuffer( id ).bufferNotification( notif );
      }
    }
   
    // let all listeners know...
    sendNotification( notif );
  }

    private NotificationBuffer
  getBuffer( final Object  bufferID )
  {
    return( (NotificationBuffer)mBuffers.get( bufferID ) );
  }
 
  private final class OverflowHandlerImpl implements OverflowHandler
  {
    public OverflowHandlerImpl()  {}
   
      public void
    handleBufferOverflow( Object o )
    {
      final Notification notif  = (Notification)o;
      issueBufferOverflowNotification( notif );
    }
  }
 
  private static int  sBufferID  = 0;
    public Object
  createBuffer(
    final int          bufferSize,
    final NotificationFilter  filter )
  {
    final OverflowHandler  handler  = new OverflowHandlerImpl();
   
    final NotificationBuffer  buffer  =
      new NotificationBuffer( bufferSize, filter, handler );
   
    Object  id  = null;
   
    synchronized( mBuffers )
    {
      ++sBufferID;
      id  = "" + sBufferID;
     
      mBuffers.put( id, buffer );
    }
   
    return( id );
  }
 
    public void
  removeBuffer( final Object  bufferID )
  {
    final NotificationBuffer  buffer  =
      (NotificationBuffer)mBuffers.remove( bufferID );
  }

    public Map
  getBufferNotifications(
    final Object  bufferID,
    final long    sequenceNumberIn )
  {
    final NotificationBuffer  buffer  = getBuffer( bufferID );
   
    return( buffer.getNotifications( sequenceNumberIn ) );
  }
 
 
    public String
  getGroup()
  {
    return( AMX.GROUP_UTILITY );
  }
 
    protected Set<ObjectName>
  getMatchingObjectNames( final ObjectName  pattern )
  {
    Set<ObjectName>  s  = null;
   
    if ( pattern.isPattern() )
    {
      s  = JMXUtil.queryNames( getMBeanServer(), pattern, null );
    }
    else
    {
      s  = GSetUtil.newSingletonSet( pattern );
    }
   
    return( s );
  }
 
    protected void
  listenToIfMatch( final ObjectName objectName )
  {
    if ( ! mListenees.keySet().contains( objectName ) )
    {
      final String  defaultDomain  = getMBeanServer().getDefaultDomain();
     
      for( final ObjectName pattern : mIncludePatterns.keySet() )
      {
        if ( JMXUtil.matchesPattern( defaultDomain, pattern, objectName ) )
        {
          final NotificationFilter  filter  =
            (NotificationFilter)mIncludePatterns.get( pattern );
       
          listenToSingle( objectName, filter );
        }
      }
    }
  }
 
    protected void
  listenToSingle(
    final ObjectName      objectName,
    final NotificationFilter  filter )
  {
    mListenees.put( objectName, filter );
    try
    {
      getMBeanServer().addNotificationListener( objectName, this, filter, null );
    }
    catch( Exception e )
    {
      mListenees.remove( objectName );
    }
  }
 
    public void
  listenTo(
    final ObjectName      pattern,
    final NotificationFilter  filter )
    throws InstanceNotFoundException
  {
    mIncludePatterns.put( pattern, filter );
   
    final Set<ObjectName>  listenees  = getMatchingObjectNames( pattern );
   
    final MBeanServer  server  = getMBeanServer();
    for( final ObjectName objectName : listenees )
    {
      if ( objectName.equals( getObjectName() ) )
      {
        continue;
      }
     
      listenToSingle( objectName, filter );
    }
  }
 
    private void
  checkListeningTo( final ObjectName objectName )
  {
    if ( ! mListenees.containsKey( objectName ) )
    {
      throw new IllegalArgumentException( objectName.toString() );
    }
  }
 
    public void
  dontListenTo( final ObjectName pattern )
  {
    mIncludePatterns.remove( pattern );
   
    final Set<ObjectName>  listenees  = getMatchingObjectNames( pattern );
   
    final MBeanServer  server  = getMBeanServer();
    for( final ObjectName objectName : listenees )
    {
      try
      {
        server.removeNotificationListener( objectName, this );
        mListenees.remove( objectName );
      }
      catch( ListenerNotFoundException e )
      {
      }
      catch( InstanceNotFoundException e )
      {
      }
    }
  }
 
 
    public Set<ObjectName>
  getListeneeObjectNameSet()
  {
    final Set<ObjectName>  objectNames  = new HashSet<ObjectName>();
   
    synchronized( mListenees )
    {
      objectNames.addAll( mListenees.keySet() );
    }
   
    return( objectNames );
  }
 
    private NotificationFilter
  _getFilter( final ObjectName objectName)
  {
    final NotificationFilter filter  =
      (NotificationFilter)mListenees.get( objectName );
   
    return( filter );
  }
 
    public NotificationFilter
  getFilter( final ObjectName objectName)
  {
    checkListeningTo( objectName );
   
    return( _getFilter( objectName ) );
  }
 
  // tracks coming and going of MBeans being listened to which
  // match our patterns...
  private final class RegistrationListener implements NotificationListener
  {
    public RegistrationListener()  {}
   
      public void
    handleNotification(
      final Notification  notifIn,
      final Object    handback)
    {
      if ( notifIn instanceof MBeanServerNotification )
      {
        final MBeanServerNotification  notif  = (MBeanServerNotification)notifIn;
       
        final ObjectName  objectName  = notif.getMBeanName();
        final String  type  = notif.getType();
       
        if ( type.equals( MBeanServerNotification.REGISTRATION_NOTIFICATION  ) )
        {
          listenToIfMatch( objectName );
        }
        else if ( type.equals( MBeanServerNotification.UNREGISTRATION_NOTIFICATION  ) )
        {
          dontListenTo( objectName );
        }
      }
     
    }
  }
 
    protected void
  preRegisterDone()
        throws Exception
  {
    super.preRegisterDone();
   
    // it's crucial we listen for registration/unregistration events
    // so that any patterns are maintained.
    JMXUtil.listenToMBeanServerDelegate( getMBeanServer(),
      new RegistrationListener(), null, null );
  }

    public void
  preDeregisterHook()
  {
    super.preDeregisterHook();
   
    synchronized( mListenees )
    {
      final Set<ObjectName> s  = getListeneeObjectNameSet();
     
      final ObjectName[]  objectNames  = new ObjectName[ s.size() ];
      s.toArray( objectNames );
     
      for( int i = 0; i < objectNames.length; ++i )
      {
        dontListenTo( objectNames[ i ] );
      }
    }
   
    synchronized( mBuffers )
    {
      for( final NotificationBuffer buffer : mBuffers.values() )
      {
        removeBuffer( buffer );
      }
    }
  }
}










TOP

Related Classes of com.sun.enterprise.management.support.NotificationServiceImpl$RegistrationListener

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.