Open
Description
I'm stuck by several days on this. The digest is correct but when I verify the xml generated, I got the signature is invalid
here's my code
const soap = require('soap');
const fs = require('fs');
const JSZip = require("jszip");
const SignedXml = require('xml-crypto').SignedXml;
const select = require('xml-crypto').xpath;
const dom = require('xmldom').DOMParser;
const forge = require('node-forge');
const pki = forge.pki;
function KeyInfoProvider(certificatePEM) {
if (!this instanceof KeyInfoProvider) {
return new KeyInfoProvider();
}
if (Buffer.isBuffer(certificatePEM)) {
certificatePEM = certificatePEM.toString('ascii');
}
if (certificatePEM == null || typeof certificatePEM !== 'string') {
throw new Error('certificatePEM must be a valid certificate in PEM format');
}
this._certificatePEM = certificatePEM;
this.getKeyInfo = function (key, prefix) {
let keyInfoXml,
certObj,
certBodyInB64;
prefix = prefix || '';
prefix = prefix ? prefix + ':' : prefix;
certBodyInB64 = forge.util.encode64(forge.pem.decode(this._certificatePEM)[1].body);
certObj = pki.certificateFromPem(`-----BEGIN CERTIFICATE-----\n${certBodyInB64}\n-----END CERTIFICATE-----`);
keyInfoXml = `<${prefix}X509Data>`;
keyInfoXml += `<${prefix}X509SubjectName>`;
keyInfoXml += `${getSubjectName(certObj)}`;
keyInfoXml += `</${prefix}X509SubjectName>`;
keyInfoXml += `<${prefix}X509Certificate>`;
keyInfoXml += `${certBodyInB64}`;
keyInfoXml += `</${prefix}X509Certificate>`;
keyInfoXml += `</${prefix}X509Data>`;
return keyInfoXml;
};
this.getKey = function () {
return this._certificatePEM;
};
};
function getSubjectName(certObj) {
const fields = ['CN', 'OU', 'O', 'L', 'ST', 'C'];
let subjectFields;
if (certObj.subject) {
subjectFields = fields.reduce(function (subjects, fieldName) {
const certAttr = certObj.subject.getField(fieldName);
if (certAttr) {
subjects.push(fieldName + '=' + certAttr.value);
}
return subjects;
}, []);
}
return Array.isArray(subjectFields) ? subjectFields.join(',') : '';
};
function normalizeLineEndings(string) {
return string.replace(/\r\n?/g, '\n').replace(/[\r\n\t ]+/g, ' ');
}
const transforms = [
'http://www.w3.org/2000/09/xmldsig#enveloped-signature',
"http://www.w3.org/2001/10/xml-exc-c14n#"
];
const digests = [
"http://www.w3.org/2001/04/xmlenc#sha256"
];
const xpath = "//*[local-name(.)='ExtensionContent']";
const options = {};
const url = 'https://e-beta.sunat.gob.pe/ol-ti-itcpfegem-beta/billService?wsdl';
const certFile = "./cert/20529473169.pem";
const data = req.param("data");
const fileName = `${data.other.companyRuc}-${data.other.documentType}-${data.other.serie}-${data.other.correlative}`;
const soapOptions = { forceSoap12Headers: false };
const certBuffer = fs.readFileSync(certFile);
let zip = new JSZip();
let sig = new SignedXml(null, options);
sig.addReference(xpath, transforms, digests);
sig.keyInfoProvider = new KeyInfoProvider(certBuffer);
sig.signingKey = certBuffer;
sig.canonicalizationAlgorithm = 'http://www.w3.org/TR/2001/REC-xml-c14n-20010315';
sig.signatureAlgorithm = "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256";
sig.computeSignature(normalizeLineEndings(zipfiles.createXML(data)), {
prefix: "ds",
attrs: {
Id: `SignatureMM`
},
location: { reference: "//*[local-name(.)='ExtensionContent']", action: "append" }
});
const xmlSigned = sig.getSignedXml();
// CHECK
let xml = xmlSigned;
const doc = new dom().parseFromString(xml);
const signature = select(doc, "//*[local-name(.)='Signature' and namespace-uri(.)='http://www.w3.org/2000/09/xmldsig#']")[0];
let sig2 = new SignedXml(null, options);
sig2.keyInfoProvider = new KeyInfoProvider(certBuffer);
sig2.loadSignature(signature);
const resxml = sig2.checkSignature(xml);
console.log("resXML", resxml);
if (!resxml) console.log(sig2.validationErrors);
// CHECK
zip.file(`${fileName}.xml`, xmlSigned);
I'm putting the generated zip too
NOTE: I was using this link too besides the company service to check the xml generated
https://tools.chilkat.io/xmlDsigVerify.cshtml
Please I really need help, I'm stuck and I don't know what to do, add normalizeLineEndings function for check if is something with the break lines, try with implicit transforms too, but nothing.