2 // SignedInfo.cs - SignedInfo implementation for XML Signature
5 // Sebastien Pouliot <sebastien@ximian.com>
6 // Tim Coleman (tim@timcoleman.com)
8 // (C) 2002, 2003 Motus Technologies Inc. (http://www.motus.com)
9 // Copyright (C) Tim Coleman, 2004
10 // (C) 2004 Novell (http://www.novell.com)
14 // Permission is hereby granted, free of charge, to any person obtaining
15 // a copy of this software and associated documentation files (the
16 // "Software"), to deal in the Software without restriction, including
17 // without limitation the rights to use, copy, modify, merge, publish,
18 // distribute, sublicense, and/or sell copies of the Software, and to
19 // permit persons to whom the Software is furnished to do so, subject to
20 // the following conditions:
22 // The above copyright notice and this permission notice shall be
23 // included in all copies or substantial portions of the Software.
25 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
26 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
27 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
28 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
29 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
30 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
31 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
34 using System
.Collections
;
37 namespace System
.Security
.Cryptography
.Xml
{
39 public class SignedInfo
: ICollection
, IEnumerable
{
41 private ArrayList references
;
42 private string c14nMethod
;
44 private string signatureMethod
;
45 private string signatureLength
;
46 private XmlElement element
;
49 XmlDsigC14NTransform canonicalizationMethodObject
;
54 references
= new ArrayList ();
55 c14nMethod
= "http://www.w3.org/TR/2001/REC-xml-c14n-20010315";
57 canonicalizationMethodObject
= new XmlDsigC14NTransform ();
61 public string CanonicalizationMethod
{
62 get { return c14nMethod; }
70 public Transform CanonicalizationMethodObject
{
71 get { return canonicalizationMethodObject; }
75 // documented as not supported (and throwing exception)
77 get { throw new NotSupportedException (); }
88 // documented as not supported (and throwing exception)
89 public bool IsReadOnly
{
90 get { throw new NotSupportedException (); }
93 // documented as not supported (and throwing exception)
94 public bool IsSynchronized
{
95 get { throw new NotSupportedException (); }
98 // Manipulating this array never affects GetXml() when
99 // LoadXml() was used.
100 // (Actually, there is no way to detect modification.)
101 public ArrayList References
{
102 get { return references; }
105 public string SignatureLength
{
106 get { return signatureLength; }
109 signatureLength
= value;
113 public string SignatureMethod
{
114 get { return signatureMethod; }
117 signatureMethod
= value;
121 // documented as not supported (and throwing exception)
122 public object SyncRoot
{
123 get { throw new NotSupportedException (); }
126 public void AddReference (Reference reference
)
128 references
.Add (reference
);
131 // documented as not supported (and throwing exception)
132 public void CopyTo (Array array
, int index
)
134 throw new NotSupportedException ();
137 public IEnumerator
GetEnumerator ()
139 return references
.GetEnumerator ();
142 public XmlElement
GetXml ()
147 if (signatureMethod
== null)
148 throw new CryptographicException ("SignatureMethod");
149 if (references
.Count
== 0)
150 throw new CryptographicException ("References empty");
152 XmlDocument document
= new XmlDocument ();
153 XmlElement xel
= document
.CreateElement (XmlSignature
.ElementNames
.SignedInfo
, XmlSignature
.NamespaceURI
);
155 xel
.SetAttribute (XmlSignature
.AttributeNames
.Id
, id
);
157 if (c14nMethod
!= null) {
158 XmlElement c14n
= document
.CreateElement (XmlSignature
.ElementNames
.CanonicalizationMethod
, XmlSignature
.NamespaceURI
);
159 c14n
.SetAttribute (XmlSignature
.AttributeNames
.Algorithm
, c14nMethod
);
160 xel
.AppendChild (c14n
);
162 if (signatureMethod
!= null) {
163 XmlElement sm
= document
.CreateElement (XmlSignature
.ElementNames
.SignatureMethod
, XmlSignature
.NamespaceURI
);
164 sm
.SetAttribute (XmlSignature
.AttributeNames
.Algorithm
, signatureMethod
);
165 if (signatureLength
!= null) {
166 XmlElement hmac
= document
.CreateElement (XmlSignature
.ElementNames
.HMACOutputLength
, XmlSignature
.NamespaceURI
);
167 hmac
.InnerText
= signatureLength
;
168 sm
.AppendChild (hmac
);
170 xel
.AppendChild (sm
);
173 // This check is only done when element is created here.
174 if (references
.Count
== 0)
175 throw new CryptographicException ("At least one Reference element is required in SignedInfo.");
177 // we add References afterward so we don't end up with extraneous
178 // xmlns="..." in each reference elements.
179 foreach (Reference r
in references
) {
180 XmlNode xn
= r
.GetXml ();
181 XmlNode newNode
= document
.ImportNode (xn
, true);
182 xel
.AppendChild (newNode
);
188 private string GetAttribute (XmlElement xel
, string attribute
)
190 XmlAttribute xa
= xel
.Attributes
[attribute
];
191 return ((xa
!= null) ? xa
.InnerText
: null);
194 public void LoadXml (XmlElement
value)
197 throw new ArgumentNullException ("value");
199 if ((value.LocalName
!= XmlSignature
.ElementNames
.SignedInfo
) || (value.NamespaceURI
!= XmlSignature
.NamespaceURI
))
200 throw new CryptographicException ();
202 id
= GetAttribute (value, XmlSignature
.AttributeNames
.Id
);
203 c14nMethod
= XmlSignature
.GetAttributeFromElement (value, XmlSignature
.AttributeNames
.Algorithm
, XmlSignature
.ElementNames
.CanonicalizationMethod
);
205 XmlElement sm
= XmlSignature
.GetChildElement (value, XmlSignature
.ElementNames
.SignatureMethod
, XmlSignature
.NamespaceURI
);
207 signatureMethod
= sm
.GetAttribute (XmlSignature
.AttributeNames
.Algorithm
);
208 XmlElement length
= XmlSignature
.GetChildElement (sm
, XmlSignature
.ElementNames
.HMACOutputLength
, XmlSignature
.NamespaceURI
);
209 if (length
!= null) {
210 signatureLength
= length
.InnerText
;
214 for (int i
= 0; i
< value.ChildNodes
.Count
; i
++) {
215 XmlNode n
= value.ChildNodes
[i
];
216 if (n
.NodeType
== XmlNodeType
.Element
&&
217 n
.LocalName
== XmlSignature
.ElementNames
.Reference
&&
218 n
.NamespaceURI
== XmlSignature
.NamespaceURI
) {
219 Reference r
= new Reference ();
220 r
.LoadXml ((XmlElement
) n
);