Package org.apache.xml.security.transforms.implementations

Source Code of org.apache.xml.security.transforms.implementations.XPath2NodeFilter

/*
* Copyright  1999-2004 The Apache Software Foundation.
*
*  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.apache.xml.security.transforms.implementations;



import java.io.IOException;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;

import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.TransformerException;

import org.apache.xml.security.c14n.CanonicalizationException;
import org.apache.xml.security.c14n.InvalidCanonicalizerException;
import org.apache.xml.security.exceptions.XMLSecurityException;
import org.apache.xml.security.signature.NodeFilter;
import org.apache.xml.security.signature.XMLSignatureInput;
import org.apache.xml.security.transforms.Transform;
import org.apache.xml.security.transforms.TransformSpi;
import org.apache.xml.security.transforms.TransformationException;
import org.apache.xml.security.transforms.Transforms;
import org.apache.xml.security.transforms.params.XPath2FilterContainer;
import org.apache.xml.security.utils.CachedXPathAPIHolder;
import org.apache.xml.security.utils.CachedXPathFuncHereAPI;
import org.apache.xml.security.utils.XMLUtils;
import org.w3c.dom.DOMException;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;

/**
* Implements the <I>XML Signature XPath Filter v2.0</I>
*
* @author $Author: mullan $
* @see <A HREF="http://www.w3.org/TR/xmldsig-filter2/">XPath Filter v2.0 (TR)</A>
* @see <a HREF="http://www.w3.org/Signature/Drafts/xmldsig-xfilter2/">XPath Filter v2.0 (editors copy)</a>
*/
public class TransformXPath2Filter extends TransformSpi {

   /** {@link org.apache.commons.logging} logging facility */
//    static org.apache.commons.logging.Log log =
//        org.apache.commons.logging.LogFactory.getLog(
//                            TransformXPath2Filter.class.getName());

   /** Field implementedTransformURI */
   public static final String implementedTransformURI =
      Transforms.TRANSFORM_XPATH2FILTER;
   //J-
   // contains the type of the filter  

   // contains the node set
 
   /**
    * Method engineGetURI
    *
    * @inheritDoc
    */
   protected String engineGetURI() {
      return implementedTransformURI;
   }



   /**
    * Method enginePerformTransform
    * @inheritDoc
    * @param input
    *
    * @throws TransformationException
    */
   protected XMLSignatureInput enginePerformTransform(XMLSignatureInput input, Transform _transformObject)
           throws TransformationException {
    CachedXPathAPIHolder.setDoc(_transformObject.getElement().getOwnerDocument());
      try {
        List unionNodes=new ArrayList();
         List subtractNodes=new ArrayList();
         List intersectNodes=new ArrayList();

         CachedXPathFuncHereAPI xPathFuncHereAPI =
            new CachedXPathFuncHereAPI(CachedXPathAPIHolder.getCachedXPathAPI());

        
         Element[] xpathElements =XMLUtils.selectNodes(
                _transformObject.getElement().getFirstChild(),
                   XPath2FilterContainer.XPathFilter2NS,
                   XPath2FilterContainer._TAG_XPATH2);
         if (xpathElements.length == 0) {
            Object exArgs[] = { Transforms.TRANSFORM_XPATH2FILTER, "XPath" };

            throw new TransformationException("xml.WrongContent", exArgs);
         }

         Document inputDoc = null;
   if (input.getSubNode() != null) {  
            inputDoc = XMLUtils.getOwnerDocument(input.getSubNode());
   } else {
            inputDoc = XMLUtils.getOwnerDocument(input.getNodeSet());
   }

         for (int i = 0; i < xpathElements.length; i++) {
            Element xpathElement = xpathElements[i];
            XPath2FilterContainer xpathContainer =
               XPath2FilterContainer.newInstance(xpathElement,
                                                   input.getSourceURI());
          

            NodeList subtreeRoots = xPathFuncHereAPI.selectNodeList(inputDoc,
                                       xpathContainer.getXPathFilterTextNode(),
                                       CachedXPathFuncHereAPI.getStrFromNode(xpathContainer.getXPathFilterTextNode()),
                                       xpathContainer.getElement());
            if (xpathContainer.isIntersect()) {
                intersectNodes.add(subtreeRoots);
             } else if (xpathContainer.isSubtract()) {
               subtractNodes.add(subtreeRoots);
             } else if (xpathContainer.isUnion()) {
                unionNodes.add(subtreeRoots);
             }
         }

        
         input.addNodeFilter(new XPath2NodeFilter(unionNodes, subtractNodes,
                                                  intersectNodes));
         input.setNodeSet(true);
         return input;
      } catch (TransformerException ex) {
         throw new TransformationException("empty", ex);
      } catch (DOMException ex) {
         throw new TransformationException("empty", ex);
      } catch (CanonicalizationException ex) {
         throw new TransformationException("empty", ex);
      } catch (InvalidCanonicalizerException ex) {
         throw new TransformationException("empty", ex);
      } catch (XMLSecurityException ex) {
         throw new TransformationException("empty", ex);
      } catch (SAXException ex) {
         throw new TransformationException("empty", ex);
      } catch (IOException ex) {
         throw new TransformationException("empty", ex);
      } catch (ParserConfigurationException ex) {
         throw new TransformationException("empty", ex);
      }
   }
}

class XPath2NodeFilter implements NodeFilter {
  boolean hasUnionFilter;
  boolean hasSubtractFilter;
  boolean hasIntersectFilter;
  XPath2NodeFilter(List unionNodes, List subtractNodes,
      List intersectNodes) {
    hasUnionFilter=!unionNodes.isEmpty();
    this.unionNodes=convertNodeListToSet(unionNodes);
    hasSubtractFilter=!subtractNodes.isEmpty();
    this.subtractNodes=convertNodeListToSet(subtractNodes);
    hasIntersectFilter=!intersectNodes.isEmpty();
    this.intersectNodes=convertNodeListToSet(intersectNodes);
  }
  Set unionNodes;
  Set subtractNodes;
  Set intersectNodes;


   /**
    * @see org.apache.xml.security.signature.NodeFilter#isNodeInclude(org.w3c.dom.Node)
    */
   public int isNodeInclude(Node currentNode) {  
     int result=1;
    
     if (hasSubtractFilter && rooted(currentNode, subtractNodes)) {
          result = -1;
     } else if (hasIntersectFilter && !rooted(currentNode, intersectNodes)) {
       result = 0;
     }
         
    //TODO OPTIMIZE
      if (result==1)              
        return 1;
      if (hasUnionFilter) {
        if (rooted(currentNode, unionNodes)) {
       return 1;
        }
        result=0;
      }     
      return result;

   }
   int inSubtract=-1;
   int inIntersect=-1;
   int inUnion=-1;
   public int isNodeIncludeDO(Node n, int level) {
     int result=1;
     if (hasSubtractFilter) {
       if ((inSubtract==-1) || (level<=inSubtract)) {
         if (inList(n,  subtractNodes)) {
           inSubtract=level;
         } else {
           inSubtract=-1;           
         }      
       }
       if (inSubtract!=-1){
         result=-1;
       }
     }
     if (result!=-1){
       if (hasIntersectFilter) {
       if ((inIntersect==-1) || (level<=inIntersect)) {
         if (!inList(n,  intersectNodes)) {
           inIntersect=-1;
           result=0;
         } else {
           inIntersect=level;           
         }      
       }
       }
     }
         
    if (level<=inUnion)
       inUnion=-1;
      if (result==1)              
        return 1;
      if (hasUnionFilter) {
        if ((inUnion==-1) && inList(n,  unionNodes)) {
          inUnion=level;
        }
        if (inUnion!=-1)
          return 1;
        result=0;
      }
         
      return result;
   }

   /**
    * Method rooted
    * @param currentNode
    * @param nodeList
    *
    * @return if rooted bye the rootnodes
    */
   static boolean  rooted(Node currentNode, Set nodeList ) {
           if (nodeList.isEmpty()) {
               return false;
           }
     if (nodeList.contains(currentNode)) {
       return true;
     }
     Iterator it=nodeList.iterator();
     while (it.hasNext()) {
         Node rootNode = (Node) it.next();
      if (XMLUtils.isDescendantOrSelf(rootNode,currentNode)) {
           return true;
      }
     }
     return false;
   }
  
      /**
       * Method rooted
       * @param currentNode
       * @param nodeList
       *
       * @return if rooted bye the rootnodes
       */
      static boolean  inList(Node currentNode, Set nodeList ) {
           return nodeList.contains(currentNode);
      }

   private static Set convertNodeListToSet(List l){
     Set result=new HashSet();
     for (int j=0;j<l.size();j++) {
       NodeList rootNodes=(NodeList) l.get(j);    
         int length = rootNodes.getLength();

         for (int i = 0; i < length; i++) {
              Node rootNode = rootNodes.item(i);
              result.add(rootNode);
             
           }
          
     }
     return result;
   }
}
TOP

Related Classes of org.apache.xml.security.transforms.implementations.XPath2NodeFilter

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.