| 
 | 
 | 
 | 
 | 
 | 
 | 
 */  | 
 | 
/**  | 
 | 
 * 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.  | 
 | 
 */  | 
 | 
/*  | 
 | 
 * Copyright (c) 2005, 2018, Oracle and/or its affiliates. All rights reserved.  | 
 | 
 */  | 
 | 
 | 
 | 
 | 
 | 
 */  | 
 | 
package org.jcp.xml.dsig.internal.dom;  | 
 | 
 | 
 | 
import java.security.Provider;  | 
 | 
import java.util.ArrayList;  | 
 | 
import java.util.Collections;  | 
 | 
import java.util.List;  | 
 | 
 | 
 | 
import javax.xml.crypto.MarshalException;  | 
 | 
import javax.xml.crypto.XMLCryptoContext;  | 
 | 
import javax.xml.crypto.XMLStructure;  | 
 | 
import javax.xml.crypto.dsig.XMLSignature;  | 
 | 
import javax.xml.crypto.dsig.dom.DOMSignContext;  | 
 | 
import javax.xml.crypto.dsig.keyinfo.KeyInfo;  | 
 | 
 | 
 | 
import org.w3c.dom.Element;  | 
 | 
import org.w3c.dom.Node;  | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 */  | 
 | 
public final class DOMKeyInfo extends BaseStructure implements KeyInfo { | 
 | 
 | 
 | 
    private final String id;  | 
 | 
    private final List<XMLStructure> keyInfoTypes;  | 
 | 
 | 
 | 
      | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
     */  | 
 | 
    @SuppressWarnings("unchecked") | 
 | 
    public static List<XMLStructure> getContent(KeyInfo ki) { | 
 | 
        return ki.getContent();  | 
 | 
    }  | 
 | 
 | 
 | 
      | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
     */  | 
 | 
    public DOMKeyInfo(List<? extends XMLStructure> content, String id) { | 
 | 
        if (content == null) { | 
 | 
            throw new NullPointerException("content cannot be null"); | 
 | 
        }  | 
 | 
        this.keyInfoTypes =  | 
 | 
            Collections.unmodifiableList(new ArrayList<>(content));  | 
 | 
        if (this.keyInfoTypes.isEmpty()) { | 
 | 
            throw new IllegalArgumentException("content cannot be empty"); | 
 | 
        }  | 
 | 
        for (int i = 0, size = this.keyInfoTypes.size(); i < size; i++) { | 
 | 
            if (!(this.keyInfoTypes.get(i) instanceof XMLStructure)) { | 
 | 
                throw new ClassCastException  | 
 | 
                    ("content["+i+"] is not a valid KeyInfo type"); | 
 | 
            }  | 
 | 
        }  | 
 | 
        this.id = id;  | 
 | 
    }  | 
 | 
 | 
 | 
      | 
 | 
 | 
 | 
 | 
 | 
 | 
 | 
     */  | 
 | 
    public DOMKeyInfo(Element kiElem, XMLCryptoContext context,  | 
 | 
                      Provider provider)  | 
 | 
        throws MarshalException  | 
 | 
    { | 
 | 
        id = DOMUtils.getIdAttributeValue(kiElem, "Id");  | 
 | 
 | 
 | 
          | 
 | 
        List<XMLStructure> content = new ArrayList<>();  | 
 | 
        Node firstChild = kiElem.getFirstChild();  | 
 | 
        if (firstChild == null) { | 
 | 
            throw new MarshalException("KeyInfo must contain at least one type"); | 
 | 
        }  | 
 | 
        while (firstChild != null) { | 
 | 
            if (firstChild.getNodeType() == Node.ELEMENT_NODE) { | 
 | 
                Element childElem = (Element)firstChild;  | 
 | 
                String localName = childElem.getLocalName();  | 
 | 
                String namespace = childElem.getNamespaceURI();  | 
 | 
                if ("X509Data".equals(localName) && XMLSignature.XMLNS.equals(namespace)) { | 
 | 
                    content.add(new DOMX509Data(childElem));  | 
 | 
                } else if ("KeyName".equals(localName) && XMLSignature.XMLNS.equals(namespace)) { | 
 | 
                    content.add(new DOMKeyName(childElem));  | 
 | 
                } else if ("KeyValue".equals(localName) && XMLSignature.XMLNS.equals(namespace)) { | 
 | 
                    content.add(DOMKeyValue.unmarshal(childElem));  | 
 | 
                } else if ("RetrievalMethod".equals(localName) && XMLSignature.XMLNS.equals(namespace)) { | 
 | 
                    content.add(new DOMRetrievalMethod(childElem,  | 
 | 
                                                       context, provider));  | 
 | 
                } else if ("PGPData".equals(localName) && XMLSignature.XMLNS.equals(namespace)) { | 
 | 
                    content.add(new DOMPGPData(childElem));  | 
 | 
                } else {  | 
 | 
                    content.add(new javax.xml.crypto.dom.DOMStructure(childElem));  | 
 | 
                }  | 
 | 
            }  | 
 | 
            firstChild = firstChild.getNextSibling();  | 
 | 
        }  | 
 | 
        keyInfoTypes = Collections.unmodifiableList(content);  | 
 | 
    }  | 
 | 
 | 
 | 
    @Override  | 
 | 
    public String getId() { | 
 | 
        return id;  | 
 | 
    }  | 
 | 
 | 
 | 
    @Override  | 
 | 
    public List<XMLStructure> getContent() { | 
 | 
        return keyInfoTypes;  | 
 | 
    }  | 
 | 
 | 
 | 
    @Override  | 
 | 
    public void marshal(XMLStructure parent, XMLCryptoContext context)  | 
 | 
        throws MarshalException  | 
 | 
    { | 
 | 
        if (parent == null) { | 
 | 
            throw new NullPointerException("parent is null"); | 
 | 
        }  | 
 | 
        if (!(parent instanceof javax.xml.crypto.dom.DOMStructure)) { | 
 | 
            throw new ClassCastException("parent must be of type DOMStructure"); | 
 | 
        }  | 
 | 
 | 
 | 
        internalMarshal( (javax.xml.crypto.dom.DOMStructure) parent, context);  | 
 | 
    }  | 
 | 
 | 
 | 
    private void internalMarshal(javax.xml.crypto.dom.DOMStructure parent, XMLCryptoContext context)  | 
 | 
            throws MarshalException { | 
 | 
        Node pNode = parent.getNode();  | 
 | 
        String dsPrefix = DOMUtils.getSignaturePrefix(context);  | 
 | 
 | 
 | 
        Node nextSibling = null;  | 
 | 
        if (context instanceof DOMSignContext) { | 
 | 
            nextSibling = ((DOMSignContext)context).getNextSibling();  | 
 | 
        }  | 
 | 
 | 
 | 
        XmlWriterToTree xwriter = new XmlWriterToTree(Marshaller.getMarshallers(), pNode, nextSibling);  | 
 | 
        marshalInternal(xwriter, this, dsPrefix, context, true);  | 
 | 
    }  | 
 | 
 | 
 | 
    public static void marshal(XmlWriter xwriter, KeyInfo ki, String dsPrefix,  | 
 | 
    XMLCryptoContext context) throws MarshalException { | 
 | 
        marshalInternal(xwriter, ki, dsPrefix, context, false);  | 
 | 
    }  | 
 | 
 | 
 | 
    private static void marshalInternal(XmlWriter xwriter, KeyInfo ki,  | 
 | 
        String dsPrefix, XMLCryptoContext context, boolean declareNamespace) throws MarshalException { | 
 | 
 | 
 | 
        xwriter.writeStartElement(dsPrefix, "KeyInfo", XMLSignature.XMLNS);  | 
 | 
        if (declareNamespace) { | 
 | 
            xwriter.writeNamespace(dsPrefix, XMLSignature.XMLNS);  | 
 | 
        }  | 
 | 
 | 
 | 
        xwriter.writeIdAttribute("", "", "Id", ki.getId()); | 
 | 
          | 
 | 
        List<XMLStructure> keyInfoTypes = getContent(ki);  | 
 | 
        for (XMLStructure kiType : keyInfoTypes) { | 
 | 
            xwriter.marshalStructure(kiType, dsPrefix, context);  | 
 | 
        }  | 
 | 
 | 
 | 
        xwriter.writeEndElement();   | 
 | 
    }  | 
 | 
 | 
 | 
    @Override  | 
 | 
    public boolean equals(Object o) { | 
 | 
        if (this == o) { | 
 | 
            return true;  | 
 | 
        }  | 
 | 
 | 
 | 
        if (!(o instanceof KeyInfo)) { | 
 | 
            return false;  | 
 | 
        }  | 
 | 
        KeyInfo oki = (KeyInfo)o;  | 
 | 
 | 
 | 
        boolean idsEqual = id == null ? oki.getId() == null  | 
 | 
                                       : id.equals(oki.getId());  | 
 | 
 | 
 | 
        return keyInfoTypes.equals(oki.getContent()) && idsEqual;  | 
 | 
    }  | 
 | 
 | 
 | 
    @Override  | 
 | 
    public int hashCode() { | 
 | 
        int result = 17;  | 
 | 
        if (id != null) { | 
 | 
            result = 31 * result + id.hashCode();  | 
 | 
        }  | 
 | 
        result = 31 * result + keyInfoTypes.hashCode();  | 
 | 
 | 
 | 
        return result;  | 
 | 
    }  | 
 | 
}  |