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

org.apache.myfaces.trinidadinternal.util.TokenCacheDebugUtils Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.apache.myfaces.trinidadinternal.util;

import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import javax.faces.component.UIViewRoot;
import javax.faces.context.ExternalContext;
import javax.faces.context.FacesContext;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;

import org.apache.myfaces.trinidad.context.RequestContext;
import org.apache.myfaces.trinidad.context.Window;
import org.apache.myfaces.trinidad.context.WindowManager;
    
 /**
  * ViewExpiredExceptions are fairly common, and the token cache is used for 
  * page state tokens, but the tokens aren't really human readable. 
  * In order to make it easier to understand what is in the cache
  * we've added a system property for debugging purposes. When enabled
  * we store a map of token -> viewId on the session which we use
  * to log something more human readable.
  * 
  * in order to use this the tester would set the system property to:
  * -Dorg.apache.myfaces.trinidadinternal.DEBUG_TOKEN_CACHE=true
  * 
  * TokenCacheDebugUtils provides methods to log human readable debugging info.
  * 
  * 
  * The goal is to provide as much logging information in a single log message as possible. 
  * In order to do this we are pushing log info into a buffer and when we're ready to log 
  * we can get the string. To start the log call like so:
  *        TokenCacheDebugUtils.startLog("My Custom String");
  * 
  * Then you can call methods like:
  *        TokenCacheDebugUtils.logCacheInfo(stateMap, null, "hello world"); 
  *        TokenCacheDebugUtils.addToLog("foo bar");
  *        
  * Then when you're ready to actually put it in the log you would do this:
  *        _LOG.severe(TokenCacheDebugUtils.getLogString());      
  */
public final class TokenCacheDebugUtils 
{
  
  private TokenCacheDebugUtils(){}


  /**
   * Checks whether we are debugging the token cache.
   * No other method in TokenCacheDebugUtils should be called unless this method returns true.
   */
  public static boolean debugTokenCache()
  {
    return _DEBUG_TOKEN_CACHE;
  }
  
  /**
   * In order to provide human readable information there is a map which 
   * has token to viewId information.
   * 
   * Add the token passed in and associate it in the map with the current view id. 
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static void addTokenToViewIdMap(String token)
  {
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);
    
    FacesContext context = FacesContext.getCurrentInstance();
    Map tokenToViewIdMap = _getTokenToViewIdMap(context);      
    UIViewRoot root = context.getViewRoot();
    String viewId = root.getViewId();    
    tokenToViewIdMap.put(token, viewId);

    StringBuffer logBuffer = _getLogBuffer(context);
    logBuffer.append("\nADDING ").append(_getTokenToViewIdString(tokenToViewIdMap, token));
  }

  /**
   * In order to provide human readable information there is a map which 
   * has token to viewId information.
   * 
   * Remove the view id info from the map for the given token.
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static void removeTokenFromViewIdMap(String token)
  {
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);

    FacesContext context = FacesContext.getCurrentInstance();
    Map tokenToViewIdMap = _getTokenToViewIdMap(context);  

    StringBuffer logBuffer = _getLogBuffer(context);
    logBuffer.append( "\nREMOVING ").append( _getTokenToViewIdString(tokenToViewIdMap, token));
    tokenToViewIdMap.remove(token);
  }  

  /**
   * Generate a string showing the token and the view id we have saved for that token
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static String getTokenToViewIdString(String token)
  {      
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);
    FacesContext context = FacesContext.getCurrentInstance();
    return _getTokenToViewIdString( _getTokenToViewIdMap(context), token);
  }


  /**
   * 
   * Add info about the cache to the log buffer.
   * 
   * For the target store map, the keys are tokens. For each token we will call 
   * getTokenToViewIdString to show what's in the map.
   * 
   * For the pinned map the keys and values are tokens, for each key/value pair call 
   * getTokenToViewIdString to show what is pinned.
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static  void logCacheInfo(Map targetStore, Map pinned, String logAddition)
  {  
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);

    FacesContext context = FacesContext.getCurrentInstance();
    Map tokenToViewId = _getTokenToViewIdMap(context);
    // TODO - add this directly to the log buffer
    StringBuffer logString =  _getLogBuffer(context);
    logString.append("\n");

    if (logAddition != null)
    {
      logString.append(logAddition).append("\n");
    }
    
    logString.append("cached token keys:");
    
    for (String targetStoreToken: targetStore.keySet()) 
    {
      logString.append("\n    ");
      logString.append(_getTokenToViewIdString(tokenToViewId, targetStoreToken));
    }  
    
    if (pinned != null)
    {
      logString.append("\n_pinned token keys:");
      
      for (String pinnedKeyToken: pinned.keySet()) 
      {
        logString.append("\n    ");
        logString.append(_getTokenToViewIdString(tokenToViewId, pinnedKeyToken));
        
        logString.append("   pinned to     ");
        String pinnedValueToken = pinned.get(pinnedKeyToken);
        logString.append(_getTokenToViewIdString(tokenToViewId, pinnedValueToken));
      }
    }    
    
    ExternalContext external = context.getExternalContext();
    Object requestObject = external.getRequest();
    
    // TODO - is there a way to get these off the external context?
    if (requestObject instanceof HttpServletRequest)
    {
      HttpServletRequest request = (HttpServletRequest)requestObject;
      logString.append("\nrequest url is " + request.getRequestURL());
      logString.append("\nrequest method is " + request.getMethod());
    }
    
    logString.append("\nrequest map entries " );    
    Iterator names = external.getRequestParameterNames();
    Map requestParams = external.getRequestParameterValuesMap();
    
    while (names.hasNext())
    {
      String name = names.next();
      String[] value = requestParams.get(name);
      
      if (value != null && value.length > 0)
      {
        logString.append("\n     " + name + " = " + value[0]);
      }
      else
      {
        logString.append("\n     " + name + " = null");
      }
    }
    
  }  


  /**
   * Add a string to the current log buffer
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static void addToLog(String addString)
  {     
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);
    
    FacesContext context = FacesContext.getCurrentInstance();
    _getLogBuffer(context).append(addString);
  }

  /**
   * Start a log buffer.
   * The startString passed in will be printed, along with session id and window id information
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static void startLog(String startString)
  {
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);
    
    FacesContext context = FacesContext.getCurrentInstance();
    StringBuffer logBuffer = _getLogBuffer(context, true);
    logBuffer.append("-------------- ").append(startString).append(" ----------\n");
    _logIdString(context);    
  }

  
  /**
   * get the string from the log buffer.
   * 
   * This method should only be called when debugTokenCache() is true
   */
  public static String getLogString()
  {
    if (!_DEBUG_TOKEN_CACHE)
      throw new UnsupportedOperationException(_UNSUPPORTED_OPERATION_MESSAGE);
    
    FacesContext context = FacesContext.getCurrentInstance();
    return _getLogBuffer(context).toString();
  }  

  private static StringBuffer _getLogBuffer(FacesContext context)
  { 
    return _getLogBuffer(context, false);
  }
  
  private static StringBuffer _getLogBuffer(FacesContext context, boolean startNewBuffer)
  { 
    Map requestMap = context.getExternalContext().getRequestMap();   
    
    StringBuffer buff = null;
    
    if (!startNewBuffer)
    { 
      buff = (StringBuffer)requestMap.get(_STRINGBUFFER_KEY);
    }
    
    if ( buff == null)
    {
      buff = new StringBuffer();
      requestMap.put(_STRINGBUFFER_KEY, buff);
    }
    
    return buff;
  }  

  private static void _logIdString(FacesContext context)
  {   
    ExternalContext externalContext = context.getExternalContext();    
    String sessionId = "";
    Object session = externalContext.getSession(false);
    
    if (session instanceof HttpSession)
    {
      sessionId = ((HttpSession)session).getId();
    }      

    StringBuffer buff = _getLogBuffer(context);
    buff.append("Session Id = ").append(sessionId);
    
    WindowManager wm = RequestContext.getCurrentInstance().getWindowManager();
    if (wm != null)
    {
      Window window = wm.getCurrentWindow(externalContext);
      
      if (window != null)
      {
        buff.append("\nWindow Id = ").append(window.getId());   
      }
      else
      {
        buff.append("\nWindow Id could not be determined, window is null" ); 
      }
        
    }
    else
    {
      buff.append("\nWindow Id could not be determined, window manager null" ); 
    }
  }  
    
  private static Map _getTokenToViewIdMap(FacesContext context)
  {    
    Map tokenToViewId = (Map)context.getExternalContext().getSessionMap().get("org.apache.myfaces.trinidadinternal.util.TOKEN_FOR_VIEW_ID");
    
    if (tokenToViewId == null) 
    {
      tokenToViewId = new ConcurrentHashMap();
      context.getExternalContext().getSessionMap().put("org.apache.myfaces.trinidadinternal.util.TOKEN_FOR_VIEW_ID", tokenToViewId);
      
    }
    return tokenToViewId;
  }  

  private static String _getTokenToViewIdString(Map tokenToViewId, String token)
  {      
    StringBuffer tokenBuffer = new StringBuffer();
    tokenBuffer.append(token);
    tokenBuffer.append(" (");
    tokenBuffer.append(tokenToViewId.get(token));
    tokenBuffer.append(")");
    
    return tokenBuffer.toString();
  }
  
  private static final String _UNSUPPORTED_OPERATION_MESSAGE =  
           "Methods in TokenCacheDebugUtils can only be called when " +
           "TokenCacheDebugUtils.debugTokenCache() returns true. " + 
           "TokenCacheDebugUtils.debugTokenCache() returns true when the system property " +
           "'org.apache.myfaces.trinidadinternal.DEBUG_TOKEN_CACHE' is true";


  private static final String _STRINGBUFFER_KEY = TokenCacheDebugUtils.class.getName() + "#StringBuffer";
    

  static private final boolean _DEBUG_TOKEN_CACHE;
  static
  {
    String dtcProp = System.getProperty("org.apache.myfaces.trinidadinternal.DEBUG_TOKEN_CACHE");
    _DEBUG_TOKEN_CACHE = Boolean.valueOf(dtcProp);  
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy