3c1f479b9d
Former-commit-id: 806294f5ded97629b74c85c09952f2a74fe182d9
220 lines
6.7 KiB
C#
220 lines
6.7 KiB
C#
//
|
|
// SignedInfo.cs - SignedInfo implementation for XML Signature
|
|
//
|
|
// Author:
|
|
// Sebastien Pouliot <sebastien@ximian.com>
|
|
// Tim Coleman (tim@timcoleman.com)
|
|
//
|
|
// (C) 2002, 2003 Motus Technologies Inc. (http://www.motus.com)
|
|
// Copyright (C) Tim Coleman, 2004
|
|
// Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
using System.Collections;
|
|
using System.Runtime.InteropServices;
|
|
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;
|
|
private XmlElement element;
|
|
|
|
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;
|
|
element = null;
|
|
}
|
|
}
|
|
|
|
[ComVisible (false)]
|
|
[MonoTODO]
|
|
public Transform CanonicalizationMethodObject {
|
|
get { throw new NotImplementedException (); }
|
|
}
|
|
|
|
// documented as not supported (and throwing exception)
|
|
public int Count {
|
|
get { throw new NotSupportedException (); }
|
|
}
|
|
|
|
public string Id {
|
|
get { return id; }
|
|
set {
|
|
element = null;
|
|
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 (); }
|
|
}
|
|
|
|
// Manipulating this array never affects GetXml() when
|
|
// LoadXml() was used.
|
|
// (Actually, there is no way to detect modification.)
|
|
public ArrayList References {
|
|
get { return references; }
|
|
}
|
|
|
|
public string SignatureLength {
|
|
get { return signatureLength; }
|
|
set {
|
|
element = null;
|
|
signatureLength = value;
|
|
}
|
|
}
|
|
|
|
public string SignatureMethod {
|
|
get { return signatureMethod; }
|
|
set {
|
|
element = null;
|
|
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 (element != null)
|
|
return element;
|
|
|
|
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);
|
|
}
|
|
|
|
// This check is only done when element is created here.
|
|
if (references.Count == 0)
|
|
throw new CryptographicException ("At least one Reference element is required in SignedInfo.");
|
|
|
|
// 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);
|
|
}
|
|
}
|
|
element = value;
|
|
}
|
|
}
|
|
}
|