"Малоформатный ссылочный элемент" при добавлении ссылки на основе атрибута Id с классом SignedXml
Невозможно подписать элемент атрибутом id при наличии префикса пространства имен:
void Main()
{
var doc = new XmlDocument();
doc.LoadXml("<root xmlns:u=\"myuri\"><test u:Id=\"_0\">Zebra</test></root>");
SignedXml signedXml = new SignedXml(doc);
signedXml.SigningKey = new RSACryptoServiceProvider();
Reference reference = new Reference("#_0");
signedXml.AddReference(reference);
signedXml.ComputeSignature();
}
ComputeSignature()
здесь не будет работать с "Malformed Reference Element", как это сделать?
Ответы
Ответ 1
Подход, который мы использовали, заключался в подклассе класса System.Security.Cryptography.Xml.SignedXml
...
public class SignedXmlWithId : SignedXml
{
public SignedXmlWithId(XmlDocument xml) : base(xml)
{
}
public SignedXmlWithId(XmlElement xmlElement)
: base(xmlElement)
{
}
public override XmlElement GetIdElement(XmlDocument doc, string id)
{
// check to see if it a standard ID reference
XmlElement idElem = base.GetIdElement(doc, id);
if (idElem == null)
{
XmlNamespaceManager nsManager = new XmlNamespaceManager(doc.NameTable);
nsManager.AddNamespace("wsu", "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd");
idElem = doc.SelectSingleNode("//*[@wsu:Id=\"" + id + "\"]", nsManager) as XmlElement;
}
return idElem;
}
}
Ответ 2
var reference = new Reference ("");//Это будет подписывать весь документ
Ответ 3
Следует отметить, что вам нужно будет использовать объект SignedXmlWithId
вместо объекта SignedXml
, чтобы иметь возможность использовать переопределенный метод GetIdElement()
. Как только я это сделал, я смог подписать XmlElement
и обойти ошибку Malformed Reference Element.
Смотрите мое сообщение об этой теме здесь.
Ответ 4
SignedXml не распознает u: Id как допустимый XML-идентификатор, а подпись XML требует, чтобы он был XML-идентификатором.
Вы можете использовать Схему (http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd, если вы пытаетесь использовать WS-Security Id) или добавить DTD к фрагменту XML. (] > для фрагмента XML). Добавление DTD только к вашему LoadXml приведет к тому, что SignedXml распознает Id, но поскольку SOAP не разрешает DTD, не включайте DTD в свой SOAP на проводнике.