我正在与马库斯·德雷尔(Markus Dreyer)一样面临同样的问题:C# 计算 SEPA (XML) 付款文件的 SHA256 值 https://stackoverflow.com/questions/18254277/c-sharp-calculate-sha256-value-for-sepa-xml-paymentfile
根据 DFÜ 协议,我必须计算 sha256 哈希值:
- 哈希值是使用整个包含的文档(包括开始和结束标记)创建的。
- 该文档根据 Canonical XML 1.0 版进行规范化。 (http://www.w3.org/TR/2001/REC-xml-c14n-20010315 http://www.w3.org/TR/2001/REC-xml-c14n-20010315).
- 如果是附属文件,还必须根据主文件进行册封。
- SHA-256 用作哈希算法。
这是一个有效的 xml 文件示例(从财务工具导出):
<?xml version="1.0" encoding="UTF-8"?>
<conxml xmlns="urn:conxml:xsd:container.nnn.002" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="urn:conxml:xsd:container.nnn.002 container.nnn.002.xsd">
<CreDtTm>2013-08-27T07:20:25Z</CreDtTm>
<MsgPain001>
<HashValue>33E579FE7A9AF6C32C100E8578EBD63E54A2DF47C6849F7A4BC8BEA9E2794197</HashValue>
<HashAlgorithm>SHA256</HashAlgorithm>
<Document xmlns="urn:swift:xsd:$pain.001.002.02">
<pain.001.001.02>
<GrpHdr>
<MsgId>D005201308270920191</MsgId>
<CreDtTm>2013-08-27T07:20:19Z</CreDtTm>
<BtchBookg>true</BtchBookg>
<NbOfTxs>1</NbOfTxs>
<CtrlSum>0.50</CtrlSum>
<Grpg>MIXD</Grpg>
<InitgPty>
<Nm>Test</Nm>
</InitgPty>
</GrpHdr>
<PmtInf>
<PmtInfId>D005201308270920191</PmtInfId>
<PmtMtd>TRF</PmtMtd>
<PmtTpInf>
<SvcLvl>
<Cd>SEPA</Cd>
</SvcLvl>
</PmtTpInf>
<ReqdExctnDt>2013-08-27</ReqdExctnDt>
<Dbtr>
<Nm>Test</Nm>
</Dbtr>
<DbtrAcct>
<Id>
<IBAN>DE76200700000888888888</IBAN>
</Id>
</DbtrAcct>
<DbtrAgt>
<FinInstnId>
<BIC>DEUTDEHHXXX</BIC>
</FinInstnId>
</DbtrAgt>
<ChrgBr>SLEV</ChrgBr>
<CdtTrfTxInf>
<PmtId>
<EndToEndId>NOTPROVIDED</EndToEndId>
</PmtId>
<Amt>
<InstdAmt Ccy="EUR">0.50</InstdAmt>
</Amt>
<CdtrAgt>
<FinInstnId>
<BIC>DEUTDEHHXXX</BIC>
</FinInstnId>
</CdtrAgt>
<Cdtr>
<Nm>Erwin Mustermann</Nm>
</Cdtr>
<CdtrAcct>
<Id>
<IBAN>DE09200700000123456789</IBAN>
</Id>
</CdtrAcct>
<RmtInf>
<Ustrd>Sepa Test Gutschrift</Ustrd>
</RmtInf>
</CdtTrfTxInf>
</PmtInf>
</pain.001.001.02>
</Document>
</MsgPain001>
</conxml>
根据 Markus Dreyer 的解决方案,这是我的代码:
System.Text.UTF8Encoding enc = new UTF8Encoding(false);
XmlDocument doc = new XmlDocument();
doc.PreserveWhitespace = true;
doc.Load(@"path to file");
XmlNodeList list = doc.GetElementsByTagName("Document");
String s = list.Item(0).OuterXml;
MemoryStream msIn = new MemoryStream(enc.GetBytes(s));
XmlDsigC14NTransform t = new XmlDsigC14NTransform(true);
t.LoadInput(msIn);
MemoryStream ms = new MemoryStream();
ms = (MemoryStream)t.GetOutput(typeof(MemoryStream));
byte[] digest = t.GetDigestedOutput(new SHA256Managed());
String result = BitConverter.ToString(digest).Replace("-", String.Empty);
在我的计算中我得到的值:
55B2597B0688AB1A19760B542AA70AEF4F980D7BC9D6EBCF2B741F6299C661D3
但预期的是文件中的值:
33E579FE7A9AF6C32C100E8578EBD63E54A2DF47C6849F7A4BC8BEA9E2794197
你们有谁知道我缺少什么吗?
通过查看这里http://www.mobilefish.com/download/sepa_xml/pain.001.001.02.xml http://www.mobilefish.com/download/sepa_xml/pain.001.001.02.xml,看来你错过了xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
Document 节点中的命名空间(可能是因为它被 XML 解析器剥离)。我修改了一些你的代码,以便它使用using
语法,如果缺少,我添加了名称空间。现在它返回正确的哈希值。
XmlDocument doc = new XmlDocument();
doc.PreserveWhitespace = true;
doc.LoadXml(xml);
XmlNodeList list = doc.GetElementsByTagName("Document");
XmlElement node = (XmlElement)list[0];
node.SetAttribute("xmlns:xsi", "http://www.w3.org/2001/XMLSchema-instance");
string s = node.OuterXml;
// The XmlDsigC14NTransform will strip the UTF8 BOM
using (MemoryStream msIn = new MemoryStream(Encoding.UTF8.GetBytes(s)))
{
XmlDsigC14NTransform t = new XmlDsigC14NTransform(true);
t.LoadInput(msIn);
using (var hash = new SHA256Managed())
{
byte[] digest = t.GetDigestedOutput(hash);
string result = BitConverter.ToString(digest).Replace("-", String.Empty);
}
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)