mirror of
https://github.com/node-ebics/node-ebics-client.git
synced 2024-11-22 06:02:09 +00:00
d192e61d29
For prettier generation of bank letters and other bits used in examples: Expand the bank config to include: * "bankFullName" used in the INI letter's which are sent to the bank. * "bankShortName" used in the filename for the generated letters and in other places if needed. * "languageCode" used for determining which template to use for the bank (currently "en" and "de" are supported). * "storageLocation" can be used to specify a local or network path where to store downloaded files. * In bankLetter.js: Use the current script folder as output for the bank's letter in HTML format instead of the user/os homedir folder. Signed-off-by: Herman van Hazendonk <github.com@herrie.org>
259 lines
7.9 KiB
JavaScript
259 lines
7.9 KiB
JavaScript
'use strict';
|
|
|
|
const $request = require('request');
|
|
|
|
const constants = require('./consts');
|
|
const Keys = require('./keymanagers/Keys');
|
|
const defaultKeyEncryptor = require('./keymanagers/defaultKeyEncryptor');
|
|
|
|
const signer = require('./middleware/signer');
|
|
const serializer = require('./middleware/serializer');
|
|
const response = require('./middleware/response');
|
|
|
|
const stringifyKeys = (keys) => {
|
|
Object.keys(keys).map((key) => {
|
|
keys[key] = keys[key] === null ? null : keys[key].toPem();
|
|
return key;
|
|
});
|
|
|
|
return JSON.stringify(keys);
|
|
};
|
|
|
|
/**
|
|
* Keys persistent object
|
|
* @typedef {Object} KeysObject
|
|
* @property {string} A006 - PEM representation of the A006 private key
|
|
* @property {string} E002 - PEM representation of the E002 private key
|
|
* @property {string} X002 - PEM representation of the X002 private key
|
|
* @property {string} bankX002 - PEM representation of the bankX002 public key
|
|
* @property {string} bankE002 - PEM representation of the bankE002 public key
|
|
*/
|
|
|
|
/**
|
|
* Key storage implementation
|
|
* @typedef {Object} KeyStorage
|
|
* @property {(data: KeysObject):Promise<void>} write - writes the keys to storage
|
|
* @property {():Promise<KeysObject>} read - reads the keys from storage
|
|
*/
|
|
|
|
/**
|
|
* Client initialization options
|
|
* @typedef {Object} EbicClientOptions
|
|
* @property {string} url - EBICS URL provided by the bank
|
|
* @property {string} partnerId - PARTNERID provided by the bank
|
|
* @property {string} hostId - HOSTID provided by the bank
|
|
* @property {string} userId - USERID provided by the bank
|
|
* @property {string} passphrase - passphrase for keys encryption
|
|
* @property {KeyStorage} keyStorage - keyStorage implementation
|
|
* @property {object} [tracesStorage] - traces (logs) storage implementation
|
|
* @property {string} bankName - Full name of the bank to be used in the bank INI letters.
|
|
* @property {string} bankShortName - Short name of the bank to be used in folders, filenames etc.
|
|
* @property {string} languageCode - Language code to be used in the bank INI letters ("de", "en" and "fr" are currently supported).
|
|
* @property {string} storageLocation - Location where to store the files that are downloaded. This can be a network share for example.
|
|
*/
|
|
|
|
|
|
module.exports = class Client {
|
|
/**
|
|
*Creates an instance of Client.
|
|
* @param {EbicClientOptions} clientOptions
|
|
*/
|
|
constructor({
|
|
url,
|
|
partnerId,
|
|
userId,
|
|
hostId,
|
|
passphrase,
|
|
keyStorage,
|
|
tracesStorage,
|
|
bankName,
|
|
bankShortName,
|
|
languageCode,
|
|
storageLocation,
|
|
}) {
|
|
if (!url)
|
|
throw new Error('EBICS URL is required');
|
|
if (!partnerId)
|
|
throw new Error('partnerId is required');
|
|
if (!userId)
|
|
throw new Error('userId is required');
|
|
if (!hostId)
|
|
throw new Error('hostId is required');
|
|
if (!passphrase)
|
|
throw new Error('passphrase is required');
|
|
|
|
if (!keyStorage || typeof keyStorage.read !== 'function' || typeof keyStorage.write !== 'function')
|
|
throw new Error('keyStorage implementation missing or wrong');
|
|
|
|
this.url = url;
|
|
this.partnerId = partnerId;
|
|
this.userId = userId;
|
|
this.hostId = hostId;
|
|
this.keyStorage = keyStorage;
|
|
this.keyEncryptor = defaultKeyEncryptor({ passphrase });
|
|
this.tracesStorage = tracesStorage || null;
|
|
this.bankName = bankName || "Dummy Bank Full Name",
|
|
this.bankShortName = bankShortName || "BANKSHORTCODE",
|
|
this.languageCode = languageCode || "en",
|
|
this.storageLocation = storageLocation || null,
|
|
}
|
|
|
|
async send(order) {
|
|
const isInObject = ('operation' in order);
|
|
|
|
if (!isInObject) throw new Error('Operation for the order needed');
|
|
|
|
if (order.operation.toUpperCase() === constants.orderOperations.ini) return this.initialization(order);
|
|
|
|
const keys = await this.keys();
|
|
if (keys === null) throw new Error('No keys provided. Can not send the order or any other order for that matter.');
|
|
|
|
if (order.operation.toUpperCase() === constants.orderOperations.upload) return this.upload(order);
|
|
if (order.operation.toUpperCase() === constants.orderOperations.download) return this.download(order);
|
|
|
|
throw new Error('Wrong order operation provided');
|
|
}
|
|
|
|
async initialization(order) {
|
|
const keys = await this.keys();
|
|
if (keys === null) await this._generateKeys();
|
|
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.new().ofType('ORDER.INI');
|
|
const res = await this.ebicsRequest(order);
|
|
const xml = res.orderData();
|
|
|
|
const returnedTechnicalCode = res.technicalCode();
|
|
const returnedBusinessCode = res.businessCode();
|
|
|
|
return {
|
|
orderData: xml.length ? xml.toString() : xml,
|
|
orderId: res.orderId(),
|
|
|
|
technicalCode: returnedTechnicalCode,
|
|
technicalCodeSymbol: res.technicalSymbol(),
|
|
technicalCodeShortText: res.technicalShortText(returnedTechnicalCode),
|
|
technicalCodeMeaning: res.technicalMeaning(returnedTechnicalCode),
|
|
|
|
businessCode: returnedBusinessCode,
|
|
businessCodeSymbol: res.businessSymbol(returnedBusinessCode),
|
|
businessCodeShortText: res.businessShortText(returnedBusinessCode),
|
|
businessCodeMeaning: res.businessMeaning(returnedBusinessCode),
|
|
|
|
bankKeys: res.bankKeys(),
|
|
};
|
|
}
|
|
|
|
async download(order) {
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.new().ofType('ORDER.DOWNLOAD');
|
|
const res = await this.ebicsRequest(order);
|
|
|
|
order.transactionId = res.transactionId();
|
|
|
|
if (res.isSegmented() && res.isLastSegment()) {
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.connect().ofType('RECEIPT.ORDER.DOWNLOAD');
|
|
|
|
await this.ebicsRequest(order);
|
|
}
|
|
|
|
const returnedTechnicalCode = res.technicalCode();
|
|
const returnedBusinessCode = res.businessCode();
|
|
|
|
return {
|
|
orderData: res.orderData(),
|
|
orderId: res.orderId(),
|
|
|
|
technicalCode: returnedTechnicalCode,
|
|
technicalCodeSymbol: res.technicalSymbol(),
|
|
technicalCodeShortText: res.technicalShortText(returnedTechnicalCode),
|
|
technicalCodeMeaning: res.technicalMeaning(returnedTechnicalCode),
|
|
|
|
businessCode: returnedBusinessCode,
|
|
businessCodeSymbol: res.businessSymbol(returnedBusinessCode),
|
|
businessCodeShortText: res.businessShortText(returnedBusinessCode),
|
|
businessCodeMeaning: res.businessMeaning(returnedBusinessCode),
|
|
};
|
|
}
|
|
|
|
async upload(order) {
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.new().ofType('ORDER.UPLOAD');
|
|
let res = await this.ebicsRequest(order);
|
|
const transactionId = res.transactionId();
|
|
const orderId = res.orderId();
|
|
|
|
order.transactionId = transactionId;
|
|
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.connect().ofType('TRANSFER.ORDER.UPLOAD');
|
|
res = await this.ebicsRequest(order);
|
|
|
|
return [transactionId, orderId];
|
|
}
|
|
|
|
ebicsRequest(order) {
|
|
return new Promise(async (resolve, reject) => {
|
|
const { version } = order;
|
|
const keys = await this.keys();
|
|
const r = signer.version(version).sign((await serializer.use(order, this)).toXML(), keys.x());
|
|
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.label(`REQUEST.${order.orderDetails.OrderType}`).data(r).persist();
|
|
|
|
$request.post({
|
|
url: this.url,
|
|
body: r,
|
|
headers: { 'content-type': 'text/xml;charset=UTF-8' },
|
|
}, (err, res, data) => {
|
|
if (err) reject(err);
|
|
|
|
const ebicsResponse = response.version(version)(data, keys);
|
|
|
|
if (this.tracesStorage)
|
|
this.tracesStorage.label(`RESPONSE.${order.orderDetails.OrderType}`).connect().data(ebicsResponse.toXML()).persist();
|
|
|
|
resolve(ebicsResponse);
|
|
});
|
|
});
|
|
}
|
|
|
|
async signOrder(order) {
|
|
const { version } = order;
|
|
const keys = await this.keys();
|
|
return signer.version(version).sign((await serializer.use(order, this)).toXML(), keys.x());
|
|
}
|
|
|
|
async keys() {
|
|
try {
|
|
const keysString = await this._readKeys();
|
|
|
|
return new Keys(JSON.parse(this.keyEncryptor.decrypt(keysString)));
|
|
} catch (err) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
async _generateKeys() {
|
|
const keysObject = Keys.generate();
|
|
|
|
await this._writeKeys(keysObject);
|
|
}
|
|
|
|
async setBankKeys(bankKeys) {
|
|
const keysObject = await this.keys();
|
|
|
|
keysObject.setBankKeys(bankKeys);
|
|
await this._writeKeys(keysObject);
|
|
}
|
|
|
|
_readKeys() {
|
|
return this.keyStorage.read();
|
|
}
|
|
|
|
_writeKeys(keysObject) {
|
|
return this.keyStorage.write(this.keyEncryptor.encrypt(stringifyKeys(keysObject.keys)));
|
|
}
|
|
};
|