mirror of
https://github.com/node-ebics/node-ebics-client.git
synced 2025-08-17 12:55:36 +00:00
wip: monorepro
This commit is contained in:
@@ -0,0 +1,67 @@
|
||||
'use strict';
|
||||
|
||||
const js2xmlparser = require('js2xmlparser');
|
||||
|
||||
const Crypto = require('../../../crypto/Crypto');
|
||||
|
||||
const genericSerializer = require('./generic');
|
||||
|
||||
module.exports = {
|
||||
async use(order, client) {
|
||||
const keys = await client.keys();
|
||||
const ebicsAccount = {
|
||||
partnerId: client.partnerId,
|
||||
userId: client.userId,
|
||||
hostId: client.hostId,
|
||||
};
|
||||
const { orderDetails, transactionId } = order;
|
||||
const {
|
||||
rootName, xmlOptions, xmlSchema, receipt, transfer, productString,
|
||||
} = genericSerializer(client.hostId, transactionId);
|
||||
|
||||
this.productString = productString;
|
||||
this.rootName = rootName;
|
||||
this.xmlOptions = xmlOptions;
|
||||
this.xmlSchema = xmlSchema;
|
||||
this.receipt = receipt;
|
||||
this.transfer = transfer;
|
||||
|
||||
if (transactionId) return this.receipt();
|
||||
|
||||
this.xmlSchema.header = {
|
||||
'@': { authenticate: true },
|
||||
static: {
|
||||
HostID: ebicsAccount.hostId,
|
||||
Nonce: Crypto.nonce(),
|
||||
Timestamp: Crypto.timestamp(),
|
||||
PartnerID: ebicsAccount.partnerId,
|
||||
UserID: ebicsAccount.userId,
|
||||
Product: {
|
||||
'@': { Language: 'en' },
|
||||
'#': productString,
|
||||
},
|
||||
OrderDetails: orderDetails,
|
||||
BankPubKeyDigests: {
|
||||
Authentication: {
|
||||
'@': { Version: 'X002', Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256' },
|
||||
'#': Crypto.digestPublicKey(keys.bankX()),
|
||||
},
|
||||
Encryption: {
|
||||
'@': { Version: 'E002', Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256' },
|
||||
'#': Crypto.digestPublicKey(keys.bankE()),
|
||||
},
|
||||
},
|
||||
SecurityMedium: '0000',
|
||||
},
|
||||
mutable: {
|
||||
TransactionPhase: 'Initialisation',
|
||||
},
|
||||
};
|
||||
|
||||
return this;
|
||||
},
|
||||
|
||||
toXML() {
|
||||
return js2xmlparser.parse(this.rootName, this.xmlSchema, this.xmlOptions);
|
||||
},
|
||||
};
|
134
packages/ebics-client/lib/orders/H004/serializers/generic.js
Normal file
134
packages/ebics-client/lib/orders/H004/serializers/generic.js
Normal file
@@ -0,0 +1,134 @@
|
||||
'use strict';
|
||||
|
||||
const constants = require('../../../consts');
|
||||
|
||||
const rootName = 'ebicsRequest';
|
||||
const rootAttributes = {
|
||||
'xmlns:ds': 'http://www.w3.org/2000/09/xmldsig#',
|
||||
xmlns: 'urn:org:ebics:H004',
|
||||
Version: 'H004',
|
||||
Revision: '1',
|
||||
};
|
||||
const header = {};
|
||||
const authSignature = ({
|
||||
'ds:SignedInfo': {
|
||||
'ds:CanonicalizationMethod': {
|
||||
'@': {
|
||||
Algorithm:
|
||||
'http://www.w3.org/TR/2001/REC-xml-c14n-20010315',
|
||||
},
|
||||
},
|
||||
'ds:SignatureMethod': {
|
||||
'@': {
|
||||
Algorithm:
|
||||
'http://www.w3.org/2001/04/xmldsig-more#rsa-sha256',
|
||||
},
|
||||
},
|
||||
'ds:Reference': {
|
||||
'@': { URI: "#xpointer(//*[@authenticate='true'])" },
|
||||
'ds:Transforms': {
|
||||
'ds:Transform': {
|
||||
'@': {
|
||||
Algorithm:
|
||||
'http://www.w3.org/TR/2001/REC-xml-c14n-20010315',
|
||||
},
|
||||
},
|
||||
},
|
||||
'ds:DigestMethod': {
|
||||
'@': {
|
||||
Algorithm:
|
||||
'http://www.w3.org/2001/04/xmlenc#sha256',
|
||||
},
|
||||
},
|
||||
'ds:DigestValue': {},
|
||||
},
|
||||
},
|
||||
'ds:SignatureValue': {},
|
||||
});
|
||||
const body = {};
|
||||
|
||||
const xmlOptions = {
|
||||
declaration: {
|
||||
include: true,
|
||||
encoding: 'utf-8',
|
||||
},
|
||||
format: {
|
||||
doubleQuotes: true,
|
||||
indent: '',
|
||||
newline: '',
|
||||
pretty: true,
|
||||
},
|
||||
};
|
||||
|
||||
module.exports = (hostId, transactionId) => ({
|
||||
// return {
|
||||
productString: constants.productString,
|
||||
rootName,
|
||||
xmlOptions,
|
||||
xmlSchema: {
|
||||
'@': rootAttributes,
|
||||
header,
|
||||
AuthSignature: authSignature,
|
||||
body,
|
||||
},
|
||||
|
||||
receipt() {
|
||||
this.xmlSchema = {
|
||||
'@': rootAttributes,
|
||||
|
||||
header: {
|
||||
'@': { authenticate: true },
|
||||
static: {
|
||||
HostID: hostId,
|
||||
TransactionID: transactionId,
|
||||
},
|
||||
mutable: {
|
||||
TransactionPhase: 'Receipt',
|
||||
},
|
||||
},
|
||||
|
||||
AuthSignature: authSignature,
|
||||
|
||||
body: {
|
||||
TransferReceipt: {
|
||||
'@': { authenticate: true },
|
||||
ReceiptCode: 0,
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
return this;
|
||||
},
|
||||
|
||||
transfer(encryptedOrderData) {
|
||||
this.xmlSchema = {
|
||||
'@': rootAttributes,
|
||||
|
||||
header: {
|
||||
'@': { authenticate: true },
|
||||
static: {
|
||||
HostID: hostId,
|
||||
TransactionID: transactionId,
|
||||
},
|
||||
mutable: {
|
||||
TransactionPhase: 'Transfer',
|
||||
SegmentNumber: {
|
||||
'@': { lastSegment: true },
|
||||
'#': 1,
|
||||
},
|
||||
},
|
||||
},
|
||||
|
||||
AuthSignature: authSignature,
|
||||
|
||||
body: {
|
||||
DataTransfer: {
|
||||
OrderData: encryptedOrderData,
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
return this;
|
||||
},
|
||||
// };
|
||||
});
|
148
packages/ebics-client/lib/orders/H004/serializers/ini.js
Normal file
148
packages/ebics-client/lib/orders/H004/serializers/ini.js
Normal file
@@ -0,0 +1,148 @@
|
||||
'use strict';
|
||||
|
||||
const zlib = require('zlib');
|
||||
|
||||
const js2xmlparser = require('js2xmlparser');
|
||||
|
||||
const Crypto = require('../../../crypto/Crypto');
|
||||
|
||||
const genericSerializer = require('./generic');
|
||||
|
||||
const keySignature = (ebicsAccount, key, xmlOptions) => {
|
||||
const xmlOrderData = {
|
||||
'@': {
|
||||
'xmlns:ds': 'http://www.w3.org/2000/09/xmldsig#',
|
||||
xmlns: 'http://www.ebics.org/S001',
|
||||
},
|
||||
SignaturePubKeyInfo: {
|
||||
PubKeyValue: {
|
||||
'ds:RSAKeyValue': {
|
||||
'ds:Modulus': key.n().toString('base64'),
|
||||
'ds:Exponent': key.e().toString('base64'),
|
||||
},
|
||||
TimeStamp: Crypto.timestamp(),
|
||||
},
|
||||
SignatureVersion: 'A006',
|
||||
},
|
||||
PartnerID: ebicsAccount.partnerId,
|
||||
UserID: ebicsAccount.userId,
|
||||
};
|
||||
|
||||
return js2xmlparser.parse('SignaturePubKeyOrderData', xmlOrderData, xmlOptions);
|
||||
};
|
||||
const orderData = (ebicsAccount, keys, xmlOptions) => {
|
||||
const xmlOrderData = {
|
||||
'@': {
|
||||
'xmlns:ds': 'http://www.w3.org/2000/09/xmldsig#',
|
||||
xmlns: 'urn:org:ebics:H004',
|
||||
},
|
||||
AuthenticationPubKeyInfo: {
|
||||
PubKeyValue: {
|
||||
'ds:RSAKeyValue': {
|
||||
'ds:Modulus': keys.x().n().toString('base64'),
|
||||
'ds:Exponent': keys.x().e().toString('base64'),
|
||||
},
|
||||
},
|
||||
AuthenticationVersion: 'X002',
|
||||
},
|
||||
EncryptionPubKeyInfo: {
|
||||
PubKeyValue: {
|
||||
'ds:RSAKeyValue': {
|
||||
'ds:Modulus': keys.e().n().toString('base64'),
|
||||
'ds:Exponent': keys.e().e().toString('base64'),
|
||||
},
|
||||
},
|
||||
EncryptionVersion: 'E002',
|
||||
},
|
||||
PartnerID: ebicsAccount.partnerId,
|
||||
UserID: ebicsAccount.userId,
|
||||
};
|
||||
|
||||
return js2xmlparser.parse('HIARequestOrderData', xmlOrderData, xmlOptions);
|
||||
};
|
||||
const commonHeader = (ebicsAccount, orderDetails, productString) => ({
|
||||
'@': { authenticate: true },
|
||||
static: {
|
||||
HostID: ebicsAccount.hostId,
|
||||
Nonce: Crypto.nonce(),
|
||||
Timestamp: Crypto.timestamp(),
|
||||
PartnerID: ebicsAccount.partnerId,
|
||||
UserID: ebicsAccount.userId,
|
||||
Product: {
|
||||
'@': { Language: 'en' },
|
||||
'#': productString,
|
||||
},
|
||||
OrderDetails: orderDetails,
|
||||
SecurityMedium: '0000',
|
||||
},
|
||||
mutable: {},
|
||||
});
|
||||
const process = {
|
||||
INI: {
|
||||
rootName: 'ebicsUnsecuredRequest',
|
||||
header: (ebicsAccount, orderDetails, productString) => {
|
||||
const ch = commonHeader(ebicsAccount, orderDetails, productString);
|
||||
|
||||
delete ch.static.Nonce;
|
||||
delete ch.static.Timestamp;
|
||||
|
||||
return ch;
|
||||
},
|
||||
body: (ebicsAccount, keys, xmlOptions) => ({
|
||||
DataTransfer: {
|
||||
OrderData: Buffer.from(zlib.deflateSync(keySignature(ebicsAccount, keys.a(), xmlOptions))).toString('base64'),
|
||||
},
|
||||
}),
|
||||
},
|
||||
HIA: {
|
||||
rootName: 'ebicsUnsecuredRequest',
|
||||
header: (ebicsAccount, orderDetails, productString) => {
|
||||
const ch = commonHeader(ebicsAccount, orderDetails, productString);
|
||||
|
||||
delete ch.static.Nonce;
|
||||
delete ch.static.Timestamp;
|
||||
|
||||
return ch;
|
||||
},
|
||||
body: (ebicsAccount, keys, xmlOptions) => ({
|
||||
DataTransfer: {
|
||||
OrderData: Buffer.from(zlib.deflateSync(orderData(ebicsAccount, keys, xmlOptions))).toString('base64'),
|
||||
},
|
||||
}),
|
||||
},
|
||||
HPB: {
|
||||
rootName: 'ebicsNoPubKeyDigestsRequest',
|
||||
header: (ebicsAccount, orderDetails, productString) => commonHeader(ebicsAccount, orderDetails, productString),
|
||||
body: () => ({}),
|
||||
},
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
async use(order, client) {
|
||||
const keys = await client.keys();
|
||||
const { orderDetails, transactionId } = order;
|
||||
const { xmlOptions, xmlSchema, productString } = genericSerializer(client.host, transactionId);
|
||||
const orderType = orderDetails.OrderType.toUpperCase();
|
||||
const ebicsAccount = {
|
||||
partnerId: client.partnerId,
|
||||
userId: client.userId,
|
||||
hostId: client.hostId,
|
||||
};
|
||||
|
||||
this.rootName = process[orderType].rootName;
|
||||
this.xmlOptions = xmlOptions;
|
||||
this.xmlSchema = xmlSchema;
|
||||
|
||||
this.xmlSchema.header = process[orderType].header(ebicsAccount, orderDetails, productString);
|
||||
this.xmlSchema.body = process[orderType].body(ebicsAccount, keys, this.xmlOptions);
|
||||
|
||||
if (orderType !== 'HPB' && Object.prototype.hasOwnProperty.call(this.xmlSchema, 'AuthSignature'))
|
||||
delete this.xmlSchema.AuthSignature;
|
||||
|
||||
return this;
|
||||
},
|
||||
|
||||
toXML() {
|
||||
return js2xmlparser.parse(this.rootName, this.xmlSchema, this.xmlOptions);
|
||||
},
|
||||
};
|
93
packages/ebics-client/lib/orders/H004/serializers/upload.js
Normal file
93
packages/ebics-client/lib/orders/H004/serializers/upload.js
Normal file
@@ -0,0 +1,93 @@
|
||||
'use strict';
|
||||
|
||||
const zlib = require('zlib');
|
||||
const crypto = require('crypto');
|
||||
|
||||
const js2xmlparser = require('js2xmlparser');
|
||||
|
||||
const Crypto = require('../../../crypto/Crypto');
|
||||
|
||||
const downloadSerializer = require('./download');
|
||||
|
||||
const transKey = crypto.randomBytes(16);
|
||||
|
||||
const signatureValue = (document, key) => {
|
||||
const digested = Crypto.digestWithHash(document.replace(/\n|\r/g, ''));
|
||||
|
||||
return Crypto.sign(key, digested);
|
||||
};
|
||||
const orderSignature = (ebicsAccount, document, key, xmlOptions) => {
|
||||
const xmlObj = {
|
||||
'@': {
|
||||
xmlns: 'http://www.ebics.org/S001',
|
||||
'xmlns:xsi': 'http://www.w3.org/2001/XMLSchema-instance',
|
||||
'xsi:schemaLocation': 'http://www.ebics.org/S001 http://www.ebics.org/S001/ebics_signature.xsd',
|
||||
},
|
||||
OrderSignatureData: {
|
||||
SignatureVersion: 'A006',
|
||||
SignatureValue: signatureValue(document, key),
|
||||
PartnerID: ebicsAccount.partnerId,
|
||||
UserID: ebicsAccount.userId,
|
||||
},
|
||||
};
|
||||
|
||||
return js2xmlparser.parse('UserSignatureData', xmlObj, xmlOptions);
|
||||
};
|
||||
const encryptedOrderSignature = (ebicsAccount, document, transactionKey, key, xmlOptions) => {
|
||||
const dst = zlib.deflateSync(orderSignature(ebicsAccount, document, key, xmlOptions));
|
||||
const cipher = crypto.createCipheriv('aes-128-cbc', transactionKey, Buffer.from([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0])).setAutoPadding(false);
|
||||
|
||||
return Buffer.concat([cipher.update(Crypto.pad(dst)), cipher.final()]).toString('base64');
|
||||
};
|
||||
const encryptedOrderData = (document, transactionKey) => {
|
||||
const dst = zlib.deflateSync(document.replace(/\n|\r/g, ''));
|
||||
const cipher = crypto.createCipheriv('aes-128-cbc', transactionKey, Buffer.from([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0])).setAutoPadding(false);
|
||||
|
||||
return Buffer.concat([cipher.update(Crypto.pad(dst)), cipher.final()]).toString('base64');
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
async use(order, client) {
|
||||
const keys = await client.keys();
|
||||
const ebicsAccount = {
|
||||
partnerId: client.partnerId,
|
||||
userId: client.userId,
|
||||
hostId: client.hostId,
|
||||
};
|
||||
const { transactionId, document } = order;
|
||||
const {
|
||||
rootName, xmlOptions, xmlSchema, transfer,
|
||||
} = await downloadSerializer.use(order, client);
|
||||
|
||||
this.rootName = rootName;
|
||||
this.xmlOptions = xmlOptions;
|
||||
this.xmlSchema = xmlSchema;
|
||||
this.transfer = transfer;
|
||||
|
||||
if (transactionId) return this.transfer(encryptedOrderData(document, transKey));
|
||||
|
||||
this.xmlSchema.header.static.NumSegments = 1;
|
||||
this.xmlSchema.body = {
|
||||
DataTransfer: {
|
||||
DataEncryptionInfo: {
|
||||
'@': { authenticate: true },
|
||||
EncryptionPubKeyDigest: {
|
||||
'@': { Version: 'E002', Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256' },
|
||||
'#': Crypto.digestPublicKey(keys.bankE()),
|
||||
},
|
||||
TransactionKey: Crypto.publicEncrypt(keys.bankE(), transKey).toString('base64'),
|
||||
},
|
||||
SignatureData: {
|
||||
'@': { authenticate: true },
|
||||
'#': encryptedOrderSignature(ebicsAccount, document, transKey, keys.a(), this.xmlOptions),
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
return this;
|
||||
},
|
||||
|
||||
toXML() {
|
||||
return js2xmlparser.parse(this.rootName, this.xmlSchema, this.xmlOptions);
|
||||
},
|
||||
};
|
Reference in New Issue
Block a user