|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  */ | 
|  |  | 
|  | package com.sun.org.apache.xerces.internal.utils; | 
|  |  | 
|  | import com.sun.org.apache.xerces.internal.utils.XMLSecurityManager.Limit; | 
|  | import java.util.Formatter; | 
|  | import java.util.HashMap; | 
|  | import java.util.Map; | 
|  | import jdk.xml.internal.JdkConstants; | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  */ | 
|  | public final class XMLLimitAnalyzer { | 
|  |  | 
|  |      | 
|  |  | 
|  |      */ | 
|  |     public static enum NameMap { | 
|  |         ENTITY_EXPANSION_LIMIT(JdkConstants.SP_ENTITY_EXPANSION_LIMIT, JdkConstants.ENTITY_EXPANSION_LIMIT), | 
|  |         MAX_OCCUR_NODE_LIMIT(JdkConstants.SP_MAX_OCCUR_LIMIT, JdkConstants.MAX_OCCUR_LIMIT), | 
|  |         ELEMENT_ATTRIBUTE_LIMIT(JdkConstants.SP_ELEMENT_ATTRIBUTE_LIMIT, JdkConstants.ELEMENT_ATTRIBUTE_LIMIT); | 
|  |  | 
|  |         final String newName; | 
|  |         final String oldName; | 
|  |  | 
|  |         NameMap(String newName, String oldName) { | 
|  |             this.newName = newName; | 
|  |             this.oldName = oldName; | 
|  |         } | 
|  |  | 
|  |         String getOldName(String newName) { | 
|  |             if (newName.equals(this.newName)) { | 
|  |                 return oldName; | 
|  |             } | 
|  |             return null; | 
|  |         } | 
|  |     } | 
|  |  | 
|  |      | 
|  |  | 
|  |      */ | 
|  |     private final int[] values; | 
|  |      | 
|  |  | 
|  |      */ | 
|  |     private final String[] names; | 
|  |      | 
|  |  | 
|  |      */ | 
|  |     private final int[] totalValue; | 
|  |  | 
|  |      | 
|  |  | 
|  |      */ | 
|  |     private final Map<String, Integer>[] caches; | 
|  |  | 
|  |     private String entityStart, entityEnd; | 
|  |      | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     @SuppressWarnings({"rawtypes", "unchecked"}) | 
|  |     public XMLLimitAnalyzer() { | 
|  |         values = new int[Limit.values().length]; | 
|  |         totalValue = new int[Limit.values().length]; | 
|  |         names = new String[Limit.values().length]; | 
|  |         caches = new Map[Limit.values().length]; | 
|  |     } | 
|  |  | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public void addValue(Limit limit, String entityName, int value) { | 
|  |         addValue(limit.ordinal(), entityName, value); | 
|  |     } | 
|  |  | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public void addValue(int index, String entityName, int value) { | 
|  |         if (index == Limit.ENTITY_EXPANSION_LIMIT.ordinal() || | 
|  |                 index == Limit.MAX_OCCUR_NODE_LIMIT.ordinal() || | 
|  |                 index == Limit.ELEMENT_ATTRIBUTE_LIMIT.ordinal() || | 
|  |                 index == Limit.TOTAL_ENTITY_SIZE_LIMIT.ordinal() || | 
|  |                 index == Limit.ENTITY_REPLACEMENT_LIMIT.ordinal() | 
|  |                 ) { | 
|  |             totalValue[index] += value; | 
|  |             return; | 
|  |         } | 
|  |         if (index == Limit.MAX_ELEMENT_DEPTH_LIMIT.ordinal() || | 
|  |                 index == Limit.MAX_NAME_LIMIT.ordinal()) { | 
|  |             values[index] = value; | 
|  |             totalValue[index] = value; | 
|  |             return; | 
|  |         } | 
|  |  | 
|  |         Map<String, Integer> cache; | 
|  |         if (caches[index] == null) { | 
|  |             cache = new HashMap<>(10); | 
|  |             caches[index] = cache; | 
|  |         } else { | 
|  |             cache = caches[index]; | 
|  |         } | 
|  |  | 
|  |         int accumulatedValue = value; | 
|  |         if (cache.containsKey(entityName)) { | 
|  |             accumulatedValue += cache.get(entityName); | 
|  |             cache.put(entityName, accumulatedValue); | 
|  |         } else { | 
|  |             cache.put(entityName, value); | 
|  |         } | 
|  |  | 
|  |         if (accumulatedValue > values[index]) { | 
|  |             values[index] = accumulatedValue; | 
|  |             names[index] = entityName; | 
|  |         } | 
|  |  | 
|  |  | 
|  |         if (index == Limit.GENERAL_ENTITY_SIZE_LIMIT.ordinal() || | 
|  |                 index == Limit.PARAMETER_ENTITY_SIZE_LIMIT.ordinal()) { | 
|  |             totalValue[Limit.TOTAL_ENTITY_SIZE_LIMIT.ordinal()] += value; | 
|  |         } | 
|  |     } | 
|  |  | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public int getValue(Limit limit) { | 
|  |         return getValue(limit.ordinal()); | 
|  |     } | 
|  |  | 
|  |     public int getValue(int index) { | 
|  |         if (index == Limit.ENTITY_REPLACEMENT_LIMIT.ordinal()) { | 
|  |             return totalValue[index]; | 
|  |         } | 
|  |         return values[index]; | 
|  |     } | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public int getTotalValue(Limit limit) { | 
|  |         return totalValue[limit.ordinal()]; | 
|  |     } | 
|  |  | 
|  |     public int getTotalValue(int index) { | 
|  |         return totalValue[index]; | 
|  |     } | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public int getValueByIndex(int index) { | 
|  |         return values[index]; | 
|  |     } | 
|  |  | 
|  |     public void startEntity(String name) { | 
|  |         entityStart = name; | 
|  |     } | 
|  |  | 
|  |     public boolean isTracking(String name) { | 
|  |         if (entityStart == null) { | 
|  |             return false; | 
|  |         } | 
|  |         return entityStart.equals(name); | 
|  |     } | 
|  |      | 
|  |  | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public void endEntity(Limit limit, String name) { | 
|  |         entityStart = ""; | 
|  |         Map<String, Integer> cache = caches[limit.ordinal()]; | 
|  |         if (cache != null) { | 
|  |             cache.remove(name); | 
|  |         } | 
|  |     } | 
|  |  | 
|  |      | 
|  |  | 
|  |  | 
|  |      */ | 
|  |     public void reset(Limit limit) { | 
|  |         if (limit.ordinal() == Limit.TOTAL_ENTITY_SIZE_LIMIT.ordinal()) { | 
|  |             totalValue[limit.ordinal()] = 0; | 
|  |         } else if (limit.ordinal() == Limit.GENERAL_ENTITY_SIZE_LIMIT.ordinal()) { | 
|  |             names[limit.ordinal()] = null; | 
|  |             values[limit.ordinal()] = 0; | 
|  |             caches[limit.ordinal()] = null; | 
|  |             totalValue[limit.ordinal()] = 0; | 
|  |         } | 
|  |     } | 
|  |  | 
|  |     public void debugPrint(XMLSecurityManager securityManager) { | 
|  |         Formatter formatter = new Formatter(); | 
|  |         System.out.println(formatter.format("%30s %15s %15s %15s %30s", | 
|  |                 "Property","Limit","Total size","Size","Entity Name")); | 
|  |  | 
|  |         for (Limit limit : Limit.values()) { | 
|  |             formatter = new Formatter(); | 
|  |             System.out.println(formatter.format("%30s %15d %15d %15d %30s", | 
|  |                     limit.name(), | 
|  |                     securityManager.getLimit(limit), | 
|  |                     totalValue[limit.ordinal()], | 
|  |                     values[limit.ordinal()], | 
|  |                     names[limit.ordinal()])); | 
|  |         } | 
|  |     } | 
|  | } |