// // SignedInfo.cs - SignedInfo implementation for XML Signature // // Author: // Sebastien Pouliot // // (C) 2002, 2003 Motus Technologies Inc. (http://www.motus.com) // (C) 2004 Novell (http://www.novell.com) // using System.Collections; using System.Xml; namespace System.Security.Cryptography.Xml { public class SignedInfo : ICollection, IEnumerable { private ArrayList references; private string c14nMethod; private string id; private string signatureMethod; private string signatureLength; public SignedInfo() { references = new ArrayList (); c14nMethod = "http://www.w3.org/TR/2001/REC-xml-c14n-20010315"; } public string CanonicalizationMethod { get { return c14nMethod; } set { c14nMethod = value; } } // documented as not supported (and throwing exception) public int Count { get { throw new NotSupportedException (); } } public string Id { get { return id; } set { id = value; } } // documented as not supported (and throwing exception) public bool IsReadOnly { get { throw new NotSupportedException (); } } // documented as not supported (and throwing exception) public bool IsSynchronized { get { throw new NotSupportedException (); } } public ArrayList References { get { return references; } } public string SignatureLength { get { return signatureLength; } set { signatureLength = value; } } public string SignatureMethod { get { return signatureMethod; } set { signatureMethod = value; } } // documented as not supported (and throwing exception) public object SyncRoot { get { throw new NotSupportedException (); } } public void AddReference (Reference reference) { references.Add (reference); } // documented as not supported (and throwing exception) public void CopyTo (Array array, int index) { throw new NotSupportedException (); } public IEnumerator GetEnumerator () { return references.GetEnumerator (); } public XmlElement GetXml() { if (signatureMethod == null) throw new CryptographicException ("SignatureMethod"); if (references.Count == 0) throw new CryptographicException ("References empty"); XmlDocument document = new XmlDocument (); XmlElement xel = document.CreateElement (XmlSignature.ElementNames.SignedInfo, XmlSignature.NamespaceURI); if (id != null) xel.SetAttribute (XmlSignature.AttributeNames.Id, id); if (c14nMethod != null) { XmlElement c14n = document.CreateElement (XmlSignature.ElementNames.CanonicalizationMethod, XmlSignature.NamespaceURI); c14n.SetAttribute (XmlSignature.AttributeNames.Algorithm, c14nMethod); xel.AppendChild (c14n); } if (signatureMethod != null) { XmlElement sm = document.CreateElement (XmlSignature.ElementNames.SignatureMethod, XmlSignature.NamespaceURI); sm.SetAttribute (XmlSignature.AttributeNames.Algorithm, signatureMethod); if (signatureLength != null) { XmlElement hmac = document.CreateElement (XmlSignature.ElementNames.HMACOutputLength, XmlSignature.NamespaceURI); hmac.InnerText = signatureLength; sm.AppendChild (hmac); } xel.AppendChild (sm); } // we add References afterward so we don't end up with extraneous // xmlns="..." in each reference elements. foreach (Reference r in references) { XmlNode xn = r.GetXml (); XmlNode newNode = document.ImportNode (xn, true); xel.AppendChild (newNode); } return xel; } private string GetAttribute (XmlElement xel, string attribute) { XmlAttribute xa = xel.Attributes [attribute]; return ((xa != null) ? xa.InnerText : null); } public void LoadXml (XmlElement value) { if (value == null) throw new ArgumentNullException ("value"); if ((value.LocalName != XmlSignature.ElementNames.SignedInfo) || (value.NamespaceURI != XmlSignature.NamespaceURI)) throw new CryptographicException (); id = GetAttribute (value, XmlSignature.AttributeNames.Id); c14nMethod = XmlSignature.GetAttributeFromElement (value, XmlSignature.AttributeNames.Algorithm, XmlSignature.ElementNames.CanonicalizationMethod); XmlElement sm = XmlSignature.GetChildElement (value, XmlSignature.ElementNames.SignatureMethod, XmlSignature.NamespaceURI); if (sm != null) { signatureMethod = sm.GetAttribute (XmlSignature.AttributeNames.Algorithm); XmlElement length = XmlSignature.GetChildElement (sm, XmlSignature.ElementNames.HMACOutputLength, XmlSignature.NamespaceURI); if (length != null) { signatureLength = length.InnerText; } } for (int i = 0; i < value.ChildNodes.Count; i++) { XmlNode n = value.ChildNodes [i]; if (n.NodeType == XmlNodeType.Element && n.LocalName == XmlSignature.ElementNames.Reference && n.NamespaceURI == XmlSignature.NamespaceURI) { Reference r = new Reference (); r.LoadXml ((XmlElement) n); AddReference (r); } } } } }