From f8e43def2e5093c798308c13101995a562a4d9fa Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 7 May 2019 16:11:23 +0800 Subject: [PATCH 01/13] serialize tranaction pass to schema --- src/TransactionQR.ts | 20 +++++++------------- 1 file changed, 7 insertions(+), 13 deletions(-) diff --git a/src/TransactionQR.ts b/src/TransactionQR.ts index 512ee52..3d2fea8 100644 --- a/src/TransactionQR.ts +++ b/src/TransactionQR.ts @@ -31,7 +31,7 @@ export class TransactionQR extends QRCode implements QRCodeInterface { /** * Construct a Transaction Request QR Code out of the * nem2-sdk Transaction instance. - * + * * @param transaction {Transaction} * @param networkType {NetworkType} * @param chainId {string} @@ -62,25 +62,19 @@ export class TransactionQR extends QRCode implements QRCodeInterface { */ public toJSON(): string { - // take the JSON of the transaction - const txJSON = this.transaction.toJSON().transaction; - - // remove empty `signature` and `signer` fields - if (txJSON.hasOwnProperty('signature') && !txJSON.signature.length) { - delete txJSON['signature']; - } - if (txJSON.hasOwnProperty('signer') && !txJSON.signer.length) { - delete txJSON['signer']; - } + // Serialize the transaction object data. + const txSerialized = this.transaction.serialize(); const jsonSchema = { 'v': 3, 'type': this.type, 'network_id': this.networkType, 'chain_id': this.chainId, - 'data': txJSON, + 'data': { + 'payload': txSerialized + } }; - return JSON.stringify(jsonSchema); + return JSON.stringify(jsonSchema).trim(); } } \ No newline at end of file From 4b0944b33166abcc454289630a1ed8d25f013289 Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 7 May 2019 16:12:21 +0800 Subject: [PATCH 02/13] added read json QR function --- src/QRCodeGenerator.ts | 38 +++++++++++++++++++++++++++++++++++++- 1 file changed, 37 insertions(+), 1 deletion(-) diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index 545536b..4ef5a0c 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -16,12 +16,14 @@ import { NetworkType, Transaction, + TransactionMapping, } from "nem2-sdk"; import {QRCode} from 'qrcode-generator-ts'; // internal dependencies import { QRCodeInterface, + QRCodeType, AccountQR, ContactQR, ObjectQR, @@ -31,7 +33,7 @@ import { /** * Class `QRCodeGenerator` describes a NIP-7 compliant QR Code * generator (factory). - * + * * @since 0.2.0 */ export class QRCodeGenerator { @@ -73,4 +75,38 @@ export class QRCodeGenerator { ): TransactionQR { return new TransactionQR(transaction, networkType, chainId); } + + /** + * Read JSON Content from QRcode. + * @param json {string} + */ + static fromJSON(json:string) :any{ + + if (json == null || json == '') { + throw Error('QR json object is missing'); + } + + const jsonObj = JSON.parse(json || ''); + + switch(jsonObj.type) { + case QRCodeType.AddContact: { + new ContactQR(jsonObj.data.account, jsonObj.network_id, jsonObj.chainId) + } + case QRCodeType.ExportAccount: { + return new AccountQR(jsonObj.data.account,jsonObj.network_id, jsonObj.chainId) + } + case QRCodeType.RequestTransaction: { + let txMapping: Transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); + + return new TransactionQR(txMapping, jsonObj.network_id, jsonObj.chainId) + } + case QRCodeType.RequestCosignature: { + // Todo: In progress; + break; + } + case QRCodeType.ExportObject: { + new ObjectQR(jsonObj.data.object, jsonObj.network_id, jsonObj.chainId); + } + } + } } From 6c15be4056b72ac49d34b48b2c28e271bc09b10c Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 7 May 2019 17:27:42 +0800 Subject: [PATCH 03/13] added fromJson's TransactionQR test --- test/QRCodeGenerator.spec.ts | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index b5d7767..4104abe 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -32,6 +32,8 @@ import { QRCodeGenerator } from "../index"; import { ExpectedObjectBase64, } from './vectors/index'; +import { TransactionQR } from "../src/TransactionQR"; +import { QRCodeType } from "../src/QRCodeType"; describe('QRCodeGenerator -->', () => { @@ -78,4 +80,37 @@ describe('QRCodeGenerator -->', () => { expect(requestTx.toJSON()).to.have.lengthOf.below(2953); }); }); + + describe('fromJson() should', () => { + + it('Read data From TransactionQR', () => { + // Arrange: + const transfer = TransferTransaction.create( + Deadline.create(), + Address.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ), + [new Mosaic(new NamespaceId('cat.currency'), UInt64.fromUint(10000000))], + PlainMessage.create('Welcome to NEM!'), + NetworkType.MIJIN_TEST + ); + + const requestTx = QRCodeGenerator.createTransactionRequest(transfer,NetworkType.MIJIN_TEST); + const txJSON = requestTx.toJSON(); + + // Act: + const transactionObj: TransactionQR = QRCodeGenerator.fromJSON(txJSON); + + // Assert: + expect(transactionObj).to.not.be.equal(''); + expect(transactionObj.transaction.toJSON()).to.deep.equal(transfer.toJSON()); + expect(transactionObj.type).to.deep.equal(QRCodeType.RequestTransaction); + }); + + + it('Read data From AccountQR', () => {}); + it('Read data From ContactQR', () => {}); + it('Read data From ObjectQR', () => {}); + }); }); From 2fe4c034e6007b07c3d4475bc0bd1518b0f92280 Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 7 May 2019 18:34:28 +0800 Subject: [PATCH 04/13] added fromJson for ContactQR --- src/ContactQR.ts | 8 ++++--- src/QRCodeGenerator.ts | 26 ++++++++++++++++++---- test/QRCodeGenerator.spec.ts | 43 +++++++++++++++++++++++++++++++++++- 3 files changed, 69 insertions(+), 8 deletions(-) diff --git a/src/ContactQR.ts b/src/ContactQR.ts index 4f9b2e6..c2f6a8d 100644 --- a/src/ContactQR.ts +++ b/src/ContactQR.ts @@ -31,7 +31,7 @@ export class ContactQR extends QRCode implements QRCodeInterface { /** * Construct a Contact QR Code out of the * nem2-sdk Account or PublicAccount instance. - * + * * @param account {Account|PublicAccount} * @param networkType {NetworkType} * @param chainId {string} @@ -67,9 +67,11 @@ export class ContactQR extends QRCode implements QRCodeInterface { 'type': this.type, 'network_id': this.networkType, 'chain_id': this.chainId, - 'data': 'not implemented yet', + 'data': { + 'address': this.account + }, }; - return JSON.stringify(jsonSchema); + return JSON.stringify(jsonSchema).trim(); } } \ No newline at end of file diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index 4ef5a0c..ecffb32 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -17,6 +17,8 @@ import { NetworkType, Transaction, TransactionMapping, + Account, + PublicAccount } from "nem2-sdk"; import {QRCode} from 'qrcode-generator-ts'; @@ -54,7 +56,7 @@ export class QRCodeGenerator { */ public static createExportObject( object: Object, - networkType: NetworkType = NetworkType.TEST_NET, + networkType: NetworkType = NetworkType.MIJIN_TEST, chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' ): ObjectQR { return new ObjectQR(object, networkType, chainId); @@ -70,12 +72,28 @@ export class QRCodeGenerator { */ public static createTransactionRequest( transaction: Transaction, - networkType: NetworkType = NetworkType.TEST_NET, + networkType: NetworkType = NetworkType.MIJIN_TEST, chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' ): TransactionQR { return new TransactionQR(transaction, networkType, chainId); } + /** + * Create a Transaction Request QR Code from a Transaction + * instance. + * + * @param transaction {Transaction} + * @param networkType {NetworkType} + * @param chainId {string} + */ + public static createContact( + account: Account | PublicAccount, + networkType: NetworkType = NetworkType.MIJIN_TEST, + chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' + ): ContactQR { + return new ContactQR(account, networkType, chainId); + } + /** * Read JSON Content from QRcode. * @param json {string} @@ -90,7 +108,7 @@ export class QRCodeGenerator { switch(jsonObj.type) { case QRCodeType.AddContact: { - new ContactQR(jsonObj.data.account, jsonObj.network_id, jsonObj.chainId) + return new ContactQR(jsonObj.data.address, jsonObj.network_id, jsonObj.chainId) } case QRCodeType.ExportAccount: { return new AccountQR(jsonObj.data.account,jsonObj.network_id, jsonObj.chainId) @@ -105,7 +123,7 @@ export class QRCodeGenerator { break; } case QRCodeType.ExportObject: { - new ObjectQR(jsonObj.data.object, jsonObj.network_id, jsonObj.chainId); + return new ObjectQR(jsonObj.data.object, jsonObj.network_id, jsonObj.chainId); } } } diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index 4104abe..bb4e09b 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -23,6 +23,7 @@ import { UInt64, PlainMessage, NetworkType, + PublicAccount, } from 'nem2-sdk'; // internal dependencies @@ -34,6 +35,7 @@ import { } from './vectors/index'; import { TransactionQR } from "../src/TransactionQR"; import { QRCodeType } from "../src/QRCodeType"; +import { ContactQR } from "../src/ContactQR"; describe('QRCodeGenerator -->', () => { @@ -81,6 +83,26 @@ describe('QRCodeGenerator -->', () => { }); }); + describe('createContact() should', ()=> { + + it('generate correct Base64 representation for TransferTransaction', () => { + // Arrange: + const account = PublicAccount.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ); + + // Act: + const createContact = QRCodeGenerator.createContact(account); + const actualBase64 = createContact.toBase64(); + + // Assert: + expect(actualBase64).to.not.be.equal(''); + expect(actualBase64.length).to.not.be.equal(0); + expect(createContact.toJSON()).to.have.lengthOf.below(2953); + }); + }); + describe('fromJson() should', () => { it('Read data From TransactionQR', () => { @@ -108,9 +130,28 @@ describe('QRCodeGenerator -->', () => { expect(transactionObj.type).to.deep.equal(QRCodeType.RequestTransaction); }); + it.only('Read data From ContactQR', () => { + // Arrange: + const account = PublicAccount.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ); + + const createContact = QRCodeGenerator.createContact(account,NetworkType.MIJIN_TEST); + const contactJSON = createContact.toJSON(); + + + // Act: + const contactObj: ContactQR = QRCodeGenerator.fromJSON(contactJSON); + + // Assert: + expect(contactObj).to.not.be.equal(''); + expect(contactObj.account.address).to.deep.equal(account.address); + expect(contactObj.type).to.deep.equal(QRCodeType.AddContact); + }); it('Read data From AccountQR', () => {}); - it('Read data From ContactQR', () => {}); + it('Read data From ObjectQR', () => {}); }); }); From 886b310ef525febf8a98f539359aeb689c4c6fe0 Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Mon, 13 May 2019 17:39:39 +0800 Subject: [PATCH 05/13] WIP export account funtion --- package-lock.json | 4 +-- package.json | 1 + src/AccountQR.ts | 35 +++++++++++++++++++------- src/QRCode.ts | 48 +++++++++++++++++++++++++++++++++++- src/QRCodeGenerator.ts | 18 +++++++++++--- src/QRCodeInterface.ts | 27 +++++++++++++++++++- test/AccountQR.spec.ts | 26 ++++++++++--------- test/QRCodeGenerator.spec.ts | 27 ++++++++++++++++++++ 8 files changed, 158 insertions(+), 28 deletions(-) diff --git a/package-lock.json b/package-lock.json index a751c5d..ad1db38 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3816,8 +3816,8 @@ } }, "webpack-dev-server": { - "version": ">=3.1.11", - "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.3.1.tgz", + "version": "1.16.5", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-1.16.5.tgz", "integrity": "sha1-DL1fLSrI1OWTqs1clwLnu9XlmJI=", "requires": { "compression": "^1.5.2", diff --git a/package.json b/package.json index e50e960..404d2c3 100644 --- a/package.json +++ b/package.json @@ -8,6 +8,7 @@ "test": "test" }, "dependencies": { + "crypto-js": "^3.1.9-1", "nem2-sdk": "^0.11.5", "qrcode-generator-ts": "^0.0.4" }, diff --git a/src/AccountQR.ts b/src/AccountQR.ts index fcce2b1..84d56b9 100644 --- a/src/AccountQR.ts +++ b/src/AccountQR.ts @@ -17,6 +17,8 @@ import { Account, PublicAccount, NetworkType, + Wallet, + Password, } from "nem2-sdk"; // internal dependencies @@ -26,12 +28,13 @@ import { QRCodeType, QRCodeSettings, } from '../index'; +import { throwError } from "rxjs"; export class AccountQR extends QRCode implements QRCodeInterface { /** * Construct an Account QR Code out of the * nem2-sdk Account or PublicAccount instance. - * + * * @param transaction {Transaction} * @param networkType {NetworkType} * @param chainId {string} @@ -41,6 +44,11 @@ export class AccountQR extends QRCode implements QRCodeInterface { * @var {Account} */ public readonly account: Account, + /** + * The password for encryption + * @var {Password} + */ + public readonly password: Password, /** * The network type. * @var {NetworkType} @@ -62,14 +70,23 @@ export class AccountQR extends QRCode implements QRCodeInterface { */ public toJSON(): string { - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': 'not implemented yet', - }; + if (this.password == null) { + throw Error('Password is missing'); + } + + const encryption = this.AES_PBKF2_encryption(this.password, this.account.privateKey); + + const jsonSchema = { + 'v': 3, + 'type': this.type, + 'network_id': this.networkType, + 'chain_id': this.chainId, + 'data': { + 'priv_key': encryption.encrypted, + 'salt': encryption.salt, + }, + }; - return JSON.stringify(jsonSchema); + return JSON.stringify(jsonSchema); } } \ No newline at end of file diff --git a/src/QRCode.ts b/src/QRCode.ts index 18ce7fb..e0d6ec0 100644 --- a/src/QRCode.ts +++ b/src/QRCode.ts @@ -21,7 +21,10 @@ import { import { NetworkType, + Password } from 'nem2-sdk'; +import {convert,nacl_catapult} from 'nem2-library'; +import * as CryptoJS from "crypto-js"; // internal dependencies import { @@ -35,7 +38,7 @@ export abstract class QRCode implements QRCodeInterface { /** * Construct a QR Code instance out of its base64 * representation and type. - * + * * @param type {QRCodeType} * @param base64 {string} */ @@ -110,4 +113,47 @@ export abstract class QRCode implements QRCodeInterface { QRCodeSettings.MARGIN_PIXEL ); } + + public AES_PBKF2_encryption(password: Password, privateKey: string): any { + const salt = CryptoJS.lib.WordArray.random(256 / 8); + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 256 / 32, + iterations: 2000, + }); + + const hex = convert.uint8ToHex(nacl_catapult.randomBytes(16)); + + const encIv = { + iv: CryptoJS.enc.Hex.parse(hex), + }; + + const encrypted = CryptoJS.AES.encrypt(CryptoJS.enc.Hex.parse(privateKey), key, encIv); + + return { + encrypted: hex + encrypted.toString(), + salt: salt.toString(), + }; + } + + public AES_PBKF2_decryption(password: Password, json: any): string { + const encryptedData = json; + const salt = CryptoJS.enc.Hex.parse(encryptedData.salt); + const iv = CryptoJS.enc.Hex.parse(encryptedData.encrypted.substring(0, 32)); + const encrypted: string = encryptedData.encrypted.substring(32, 96); + + //generate key + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 256 / 32, + iterations: 2000, + }); + + let encIv = { + iv: iv + }; + + let decrypt = CryptoJS.enc.Hex.stringify(CryptoJS.AES.decrypt(encrypted, key, encIv)); + + if (decrypt === "" || (decrypt.length != 64 && decrypt.length != 66)) throw new Error("invalid password"); + return decrypt; + } } \ No newline at end of file diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index ecffb32..aab1b4e 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -18,7 +18,8 @@ import { Transaction, TransactionMapping, Account, - PublicAccount + PublicAccount, + Password } from "nem2-sdk"; import {QRCode} from 'qrcode-generator-ts'; @@ -94,6 +95,15 @@ export class QRCodeGenerator { return new ContactQR(account, networkType, chainId); } + public static createExportAccount( + account: Account, + password: Password, + networkType: NetworkType = NetworkType.MIJIN_TEST, + chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31', + ): AccountQR { + return new AccountQR(account, password, networkType, chainId); + } + /** * Read JSON Content from QRcode. * @param json {string} @@ -110,9 +120,9 @@ export class QRCodeGenerator { case QRCodeType.AddContact: { return new ContactQR(jsonObj.data.address, jsonObj.network_id, jsonObj.chainId) } - case QRCodeType.ExportAccount: { - return new AccountQR(jsonObj.data.account,jsonObj.network_id, jsonObj.chainId) - } + // case QRCodeType.ExportAccount: { + // return new AccountQR(jsonObj.data.account,jsonObj.network_id, jsonObj.chainId) + // } case QRCodeType.RequestTransaction: { let txMapping: Transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); diff --git a/src/QRCodeInterface.ts b/src/QRCodeInterface.ts index 8d546e8..a14a82f 100644 --- a/src/QRCodeInterface.ts +++ b/src/QRCodeInterface.ts @@ -15,13 +15,17 @@ */ import {QRCode} from 'qrcode-generator-ts'; +import { + Password +} from 'nem2-sdk'; + // internal dependencies import {QRCodeType} from '../index'; /** * Interface `QRCodeInterface` describes rules for the definition * of NIP-7 compliant QR Codes. - * + * * @since 0.2.0 */ export interface QRCodeInterface { @@ -60,4 +64,25 @@ export interface QRCodeInterface { * @return {string} */ toBase64(): string; + + /** + * The `AES_PBKF2_encryption()` method should return encrypted and salt + * + * @param password + * @param privateKey + * + * @returns {json} + */ + AES_PBKF2_encryption(password: Password, privateKey: string): any; + + /** + * The `AES_PBKF2_decryption()` method should return string (privateKey) + * + * @param password + * @param json + * + * @returns {string} + */ + AES_PBKF2_decryption(password: Password, json: any): string; + } diff --git a/test/AccountQR.spec.ts b/test/AccountQR.spec.ts index 79a973b..39a310c 100644 --- a/test/AccountQR.spec.ts +++ b/test/AccountQR.spec.ts @@ -17,6 +17,7 @@ import {expect} from "chai"; import { Account, NetworkType, + Password, } from 'nem2-sdk'; import { QRCode as QRCodeImpl, @@ -25,13 +26,14 @@ import { } from 'qrcode-generator-ts'; // internal dependencies -import { +import { QRCodeInterface, QRCode, QRCodeType, QRCodeSettings, ContactQR, } from "../index"; +import { AccountQR } from "../src/AccountQR"; describe('AccountQR -->', () => { @@ -44,17 +46,19 @@ describe('AccountQR -->', () => { NetworkType.TEST_NET ); + const password = new Password('1234'); + // Act: - const addContact = new ContactQR(account, NetworkType.TEST_NET, ''); - const actualJSON = addContact.toJSON(); - const actualObject = JSON.parse(actualJSON); - - // Assert: - expect(actualObject).to.have.property('v'); - expect(actualObject).to.have.property('type'); - expect(actualObject).to.have.property('network_id'); - expect(actualObject).to.have.property('chain_id'); - expect(actualObject).to.have.property('data'); + const exportAccount = new AccountQR(account, password, NetworkType.TEST_NET, ''); + const actualJSON = exportAccount.toJSON(); + // const actualObject = JSON.parse(actualJSON); + + // // Assert: + // expect(actualObject).to.have.property('v'); + // expect(actualObject).to.have.property('type'); + // expect(actualObject).to.have.property('network_id'); + // expect(actualObject).to.have.property('chain_id'); + // expect(actualObject).to.have.property('data'); }); }); diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index bb4e09b..4ca0052 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -24,6 +24,8 @@ import { PlainMessage, NetworkType, PublicAccount, + Account, + Password } from 'nem2-sdk'; // internal dependencies @@ -36,6 +38,7 @@ import { import { TransactionQR } from "../src/TransactionQR"; import { QRCodeType } from "../src/QRCodeType"; import { ContactQR } from "../src/ContactQR"; +import { AccountQR } from "../src/AccountQR"; describe('QRCodeGenerator -->', () => { @@ -103,6 +106,30 @@ describe('QRCodeGenerator -->', () => { }); }); + describe.only('createExportAccount() should', ()=> { + + it('generate correct Base64 representation for ExportAccount', () => { + // Arrange: + const account = Account.createFromPrivateKey( + 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', + NetworkType.TEST_NET + ); + const password = new Password('password'); + + + // Act: + const exportAccount = QRCodeGenerator.createExportAccount(account,password); + const actualObject = exportAccount.toJSON(); + // const actualBase64 = exportAccount.base64(); + console.log(exportAccount); + + // Assert: + expect(actualObject).to.not.be.equal(''); + expect(actualObject.length).to.not.be.equal(0); + expect(exportAccount.toJSON()).to.have.lengthOf.below(2953); + }); + }); + describe('fromJson() should', () => { it('Read data From TransactionQR', () => { From cc44245b9fa4fa7a522b2d32eeddafbe9b54753c Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 14 May 2019 11:29:43 +0800 Subject: [PATCH 06/13] completed generate exportAccount json --- src/AccountQR.ts | 2 +- test/AccountQR.spec.ts | 20 ++++++++++---------- test/QRCodeGenerator.spec.ts | 19 ++++++------------- 3 files changed, 17 insertions(+), 24 deletions(-) diff --git a/src/AccountQR.ts b/src/AccountQR.ts index 84d56b9..a8fc094 100644 --- a/src/AccountQR.ts +++ b/src/AccountQR.ts @@ -48,7 +48,7 @@ export class AccountQR extends QRCode implements QRCodeInterface { * The password for encryption * @var {Password} */ - public readonly password: Password, + protected readonly password: Password, /** * The network type. * @var {NetworkType} diff --git a/test/AccountQR.spec.ts b/test/AccountQR.spec.ts index 39a310c..d94e3d6 100644 --- a/test/AccountQR.spec.ts +++ b/test/AccountQR.spec.ts @@ -37,28 +37,28 @@ import { AccountQR } from "../src/AccountQR"; describe('AccountQR -->', () => { - describe('toJSON() should', () => { + describe.only('toJSON() should', () => { it('include mandatory NIP-7 QR Code base fields', () => { // Arrange: const account = Account.createFromPrivateKey( 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', - NetworkType.TEST_NET + NetworkType.MIJIN_TEST ); - const password = new Password('1234'); + const password = new Password('password'); // Act: - const exportAccount = new AccountQR(account, password, NetworkType.TEST_NET, ''); + const exportAccount = new AccountQR(account, password, NetworkType.MIJIN_TEST, ''); const actualJSON = exportAccount.toJSON(); - // const actualObject = JSON.parse(actualJSON); + const actualObject = JSON.parse(actualJSON); // // Assert: - // expect(actualObject).to.have.property('v'); - // expect(actualObject).to.have.property('type'); - // expect(actualObject).to.have.property('network_id'); - // expect(actualObject).to.have.property('chain_id'); - // expect(actualObject).to.have.property('data'); + expect(actualObject).to.have.property('v'); + expect(actualObject).to.have.property('type'); + expect(actualObject).to.have.property('network_id'); + expect(actualObject).to.have.property('chain_id'); + expect(actualObject).to.have.property('data'); }); }); diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index 4ca0052..fd5590d 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -29,16 +29,12 @@ import { } from 'nem2-sdk'; // internal dependencies -import { QRCodeGenerator } from "../index"; +import { QRCodeGenerator, AccountQR, TransactionQR, QRCodeType, ContactQR } from "../index"; // vectors data import { ExpectedObjectBase64, } from './vectors/index'; -import { TransactionQR } from "../src/TransactionQR"; -import { QRCodeType } from "../src/QRCodeType"; -import { ContactQR } from "../src/ContactQR"; -import { AccountQR } from "../src/AccountQR"; describe('QRCodeGenerator -->', () => { @@ -106,26 +102,23 @@ describe('QRCodeGenerator -->', () => { }); }); - describe.only('createExportAccount() should', ()=> { + describe('createExportAccount() should', ()=> { it('generate correct Base64 representation for ExportAccount', () => { // Arrange: const account = Account.createFromPrivateKey( 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', - NetworkType.TEST_NET + NetworkType.MIJIN_TEST ); const password = new Password('password'); - // Act: const exportAccount = QRCodeGenerator.createExportAccount(account,password); - const actualObject = exportAccount.toJSON(); - // const actualBase64 = exportAccount.base64(); - console.log(exportAccount); + const actualBase64 = exportAccount.toBase64(); // Assert: - expect(actualObject).to.not.be.equal(''); - expect(actualObject.length).to.not.be.equal(0); + expect(actualBase64).to.not.be.equal(''); + expect(actualBase64.length).to.not.be.equal(0); expect(exportAccount.toJSON()).to.have.lengthOf.below(2953); }); }); From 11a1173eb594afd7839632267b0dc60cf2237fef Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 14 May 2019 16:29:27 +0800 Subject: [PATCH 07/13] added QRService class --- index.ts | 1 + src/QRService.ts | 77 ++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 78 insertions(+) create mode 100644 src/QRService.ts diff --git a/index.ts b/index.ts index b859f30..0d2713c 100644 --- a/index.ts +++ b/index.ts @@ -22,3 +22,4 @@ export { ContactQR } from './src/ContactQR'; export { ObjectQR } from './src/ObjectQR'; export { TransactionQR } from './src/TransactionQR'; export { QRCodeGenerator } from './src/QRCodeGenerator'; +export { QRService } from './src/QRService'; diff --git a/src/QRService.ts b/src/QRService.ts new file mode 100644 index 0000000..ea3bb58 --- /dev/null +++ b/src/QRService.ts @@ -0,0 +1,77 @@ +/* + * Copyright (c) 2019 NEM Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License "); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { + Password, +} from "nem2-sdk"; +import {convert,nacl_catapult} from 'nem2-library'; +import * as CryptoJS from "crypto-js"; + +export class QRService { + + /** + * AES_PBKF2_encryption will encrypt privateKey with provided password. + * @param password {Password} + * @param privateKey {strinf} + */ + public AES_PBKF2_encryption(password: Password, privateKey: string): any { + const salt = CryptoJS.lib.WordArray.random(256 / 8); + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 256 / 32, + iterations: 2000, + }); + + const hex = convert.uint8ToHex(nacl_catapult.randomBytes(16)); + + const encIv = { + iv: CryptoJS.enc.Hex.parse(hex), + }; + + const encrypted = CryptoJS.AES.encrypt(CryptoJS.enc.Hex.parse(privateKey), key, encIv); + + return { + encrypted: hex + encrypted.toString(), + salt: salt.toString(), + }; + } + + /** + * AES_PBKF2_decryption will decrypt privateKey with provided password + * @param password + * @param json + */ + public AES_PBKF2_decryption(password: Password, json: any): string { + const encryptedData = json; + const salt = CryptoJS.enc.Hex.parse(encryptedData.data.salt); + const iv = CryptoJS.enc.Hex.parse(encryptedData.data.priv_key.substring(0, 32)); + const encrypted: string = encryptedData.data.priv_key.substring(32, 96); + + //generate key + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 256 / 32, + iterations: 2000, + }); + + let encIv = { + iv: iv + }; + + let decrypt = CryptoJS.enc.Hex.stringify(CryptoJS.AES.decrypt(encrypted, key, encIv)); + + if (decrypt === "" || (decrypt.length != 64 && decrypt.length != 66)) throw new Error("invalid password"); + return decrypt; + } +} \ No newline at end of file From 20536dc97ae32e8ba669155e2e4d07b336aebf54 Mon Sep 17 00:00:00 2001 From: Anthony Law Yong Chuan Date: Tue, 14 May 2019 16:39:28 +0800 Subject: [PATCH 08/13] password-protected export account apply --- src/AccountQR.ts | 7 +++--- src/QRCode.ts | 43 ------------------------------------ src/QRCodeGenerator.ts | 21 +++++++++++++----- src/QRCodeInterface.ts | 25 --------------------- test/AccountQR.spec.ts | 9 ++++---- test/QRCodeGenerator.spec.ts | 22 ++++++++++++++++-- 6 files changed, 44 insertions(+), 83 deletions(-) diff --git a/src/AccountQR.ts b/src/AccountQR.ts index a8fc094..078fecb 100644 --- a/src/AccountQR.ts +++ b/src/AccountQR.ts @@ -27,6 +27,7 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, + QRService } from '../index'; import { throwError } from "rxjs"; @@ -35,7 +36,7 @@ export class AccountQR extends QRCode implements QRCodeInterface { * Construct an Account QR Code out of the * nem2-sdk Account or PublicAccount instance. * - * @param transaction {Transaction} + * @param account {Account} * @param networkType {NetworkType} * @param chainId {string} */ @@ -73,8 +74,8 @@ export class AccountQR extends QRCode implements QRCodeInterface { if (this.password == null) { throw Error('Password is missing'); } - - const encryption = this.AES_PBKF2_encryption(this.password, this.account.privateKey); + const qrService: QRService = new QRService(); + const encryption = qrService.AES_PBKF2_encryption(this.password, this.account.privateKey); const jsonSchema = { 'v': 3, diff --git a/src/QRCode.ts b/src/QRCode.ts index e0d6ec0..64f432e 100644 --- a/src/QRCode.ts +++ b/src/QRCode.ts @@ -113,47 +113,4 @@ export abstract class QRCode implements QRCodeInterface { QRCodeSettings.MARGIN_PIXEL ); } - - public AES_PBKF2_encryption(password: Password, privateKey: string): any { - const salt = CryptoJS.lib.WordArray.random(256 / 8); - const key = CryptoJS.PBKDF2(password.value, salt, { - keySize: 256 / 32, - iterations: 2000, - }); - - const hex = convert.uint8ToHex(nacl_catapult.randomBytes(16)); - - const encIv = { - iv: CryptoJS.enc.Hex.parse(hex), - }; - - const encrypted = CryptoJS.AES.encrypt(CryptoJS.enc.Hex.parse(privateKey), key, encIv); - - return { - encrypted: hex + encrypted.toString(), - salt: salt.toString(), - }; - } - - public AES_PBKF2_decryption(password: Password, json: any): string { - const encryptedData = json; - const salt = CryptoJS.enc.Hex.parse(encryptedData.salt); - const iv = CryptoJS.enc.Hex.parse(encryptedData.encrypted.substring(0, 32)); - const encrypted: string = encryptedData.encrypted.substring(32, 96); - - //generate key - const key = CryptoJS.PBKDF2(password.value, salt, { - keySize: 256 / 32, - iterations: 2000, - }); - - let encIv = { - iv: iv - }; - - let decrypt = CryptoJS.enc.Hex.stringify(CryptoJS.AES.decrypt(encrypted, key, encIv)); - - if (decrypt === "" || (decrypt.length != 64 && decrypt.length != 66)) throw new Error("invalid password"); - return decrypt; - } } \ No newline at end of file diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index aab1b4e..60e8874 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -21,7 +21,7 @@ import { PublicAccount, Password } from "nem2-sdk"; -import {QRCode} from 'qrcode-generator-ts'; +import * as CryptoJS from "crypto-js"; // internal dependencies import { @@ -31,6 +31,7 @@ import { ContactQR, ObjectQR, TransactionQR, + QRService, } from '../index'; /** @@ -108,7 +109,7 @@ export class QRCodeGenerator { * Read JSON Content from QRcode. * @param json {string} */ - static fromJSON(json:string) :any{ + static fromJSON(json:string, password?: Password) :any { if (json == null || json == '') { throw Error('QR json object is missing'); @@ -120,9 +121,19 @@ export class QRCodeGenerator { case QRCodeType.AddContact: { return new ContactQR(jsonObj.data.address, jsonObj.network_id, jsonObj.chainId) } - // case QRCodeType.ExportAccount: { - // return new AccountQR(jsonObj.data.account,jsonObj.network_id, jsonObj.chainId) - // } + case QRCodeType.ExportAccount: { + if (password == null){ + throw Error('Password are required'); + } + + const qrService: QRService = new QRService(); + const privatekey: string = qrService.AES_PBKF2_decryption(password,jsonObj); + + const account = Account.createFromPrivateKey(privatekey, + NetworkType.MIJIN_TEST); + + return new AccountQR(account, password, jsonObj.network_id, jsonObj.chainId) + } case QRCodeType.RequestTransaction: { let txMapping: Transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); diff --git a/src/QRCodeInterface.ts b/src/QRCodeInterface.ts index a14a82f..8e1b110 100644 --- a/src/QRCodeInterface.ts +++ b/src/QRCodeInterface.ts @@ -15,10 +15,6 @@ */ import {QRCode} from 'qrcode-generator-ts'; -import { - Password -} from 'nem2-sdk'; - // internal dependencies import {QRCodeType} from '../index'; @@ -64,25 +60,4 @@ export interface QRCodeInterface { * @return {string} */ toBase64(): string; - - /** - * The `AES_PBKF2_encryption()` method should return encrypted and salt - * - * @param password - * @param privateKey - * - * @returns {json} - */ - AES_PBKF2_encryption(password: Password, privateKey: string): any; - - /** - * The `AES_PBKF2_decryption()` method should return string (privateKey) - * - * @param password - * @param json - * - * @returns {string} - */ - AES_PBKF2_decryption(password: Password, json: any): string; - } diff --git a/test/AccountQR.spec.ts b/test/AccountQR.spec.ts index d94e3d6..17b5787 100644 --- a/test/AccountQR.spec.ts +++ b/test/AccountQR.spec.ts @@ -37,23 +37,22 @@ import { AccountQR } from "../src/AccountQR"; describe('AccountQR -->', () => { - describe.only('toJSON() should', () => { + describe('toJSON() should', () => { - it('include mandatory NIP-7 QR Code base fields', () => { + it.only('include mandatory NIP-7 QR Code base fields', () => { // Arrange: const account = Account.createFromPrivateKey( 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', NetworkType.MIJIN_TEST ); - const password = new Password('password'); // Act: - const exportAccount = new AccountQR(account, password, NetworkType.MIJIN_TEST, ''); + const exportAccount = new AccountQR(account, password, NetworkType.MIJIN_TEST, 'no-chain-id'); const actualJSON = exportAccount.toJSON(); const actualObject = JSON.parse(actualJSON); - // // Assert: + // // // Assert: expect(actualObject).to.have.property('v'); expect(actualObject).to.have.property('type'); expect(actualObject).to.have.property('network_id'); diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index fd5590d..0677869 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -150,7 +150,7 @@ describe('QRCodeGenerator -->', () => { expect(transactionObj.type).to.deep.equal(QRCodeType.RequestTransaction); }); - it.only('Read data From ContactQR', () => { + it('Read data From ContactQR', () => { // Arrange: const account = PublicAccount.createFromPublicKey( 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', @@ -170,7 +170,25 @@ describe('QRCodeGenerator -->', () => { expect(contactObj.type).to.deep.equal(QRCodeType.AddContact); }); - it('Read data From AccountQR', () => {}); + it('Read data From AccountQR', () => { + // Arrange: + const account = Account.createFromPrivateKey( + 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', + NetworkType.MIJIN_TEST + ); + const password = new Password('password'); + + const exportAccount = QRCodeGenerator.createExportAccount(account,password); + const actualObj = exportAccount.toJSON(); + + // Act: + const accountObj: AccountQR = QRCodeGenerator.fromJSON(actualObj,password); + + // Assert: + expect(accountObj).to.not.be.equal(''); + expect(accountObj.account).to.deep.equal(account); + expect(accountObj.type).to.deep.equal(QRCodeType.ExportAccount); + }); it('Read data From ObjectQR', () => {}); }); From 8dd4dc165e66b6929fbbc6e9f266004227c89fa6 Mon Sep 17 00:00:00 2001 From: Greg S Date: Wed, 22 May 2019 13:54:35 +0200 Subject: [PATCH 09/13] nemtech/NIP#21 : refactor with Schemas implementation, fixed AccountQR, ContactQR and TransactionQR, added CosignatureQR --- index.ts | 20 ++- package-lock.json | 6 + package.json | 1 + src/AccountQR.ts | 65 ++++++---- src/ContactQR.ts | 59 ++++++--- src/CosignatureQR.ts | 101 ++++++++++++++++ src/EncryptedPayload.ts | 82 +++++++++++++ src/ObjectQR.ts | 49 ++++++-- src/QRCode.ts | 33 ++++- src/QRCodeDataSchema.ts | 67 +++++++++++ src/QRCodeGenerator.ts | 86 ++++++++----- src/QRCodeSettings.ts | 14 +-- src/TransactionQR.ts | 52 +++++--- src/schemas/AddContactDataSchema.ts | 82 +++++++++++++ src/schemas/ExportAccountDataSchema.ts | 93 ++++++++++++++ src/schemas/ExportObjectDataSchema.ts | 73 +++++++++++ src/schemas/RequestCosignatureDataSchema.ts | 74 ++++++++++++ src/schemas/RequestTransactionDataSchema.ts | 84 +++++++++++++ src/services/EncryptionService.ts | 127 ++++++++++++++++++++ test/ContactQR.spec.ts | 3 +- test/QRCode.spec.ts | 21 ++-- test/QRCodeGenerator.spec.ts | 24 +++- 22 files changed, 1084 insertions(+), 132 deletions(-) create mode 100644 src/CosignatureQR.ts create mode 100644 src/EncryptedPayload.ts create mode 100644 src/QRCodeDataSchema.ts create mode 100644 src/schemas/AddContactDataSchema.ts create mode 100644 src/schemas/ExportAccountDataSchema.ts create mode 100644 src/schemas/ExportObjectDataSchema.ts create mode 100644 src/schemas/RequestCosignatureDataSchema.ts create mode 100644 src/schemas/RequestTransactionDataSchema.ts create mode 100644 src/services/EncryptionService.ts diff --git a/index.ts b/index.ts index 0d2713c..80eb989 100644 --- a/index.ts +++ b/index.ts @@ -13,13 +13,31 @@ * See the License for the specific language governing permissions and *limitations under the License. */ + +// enumerations / interfaces export { QRCodeType } from './src/QRCodeType'; export { QRCodeSettings } from './src/QRCodeSettings'; export { QRCodeInterface } from './src/QRCodeInterface'; export { QRCode } from './src/QRCode'; + +// encryption +export { EncryptedPayload } from './src/EncryptedPayload'; +export { EncryptionService } from './src/services/EncryptionService'; + +// QR Code data schemas +export { QRCodeDataSchema } from './src/QRCodeDataSchema'; +export { AddContactDataSchema } from './src/schemas/AddContactDataSchema'; +export { ExportAccountDataSchema } from './src/schemas/ExportAccountDataSchema'; +export { ExportObjectDataSchema } from './src/schemas/ExportObjectDataSchema'; +export { RequestCosignatureDataSchema } from './src/schemas/RequestCosignatureDataSchema'; +export { RequestTransactionDataSchema } from './src/schemas/RequestTransactionDataSchema'; + +// specialized QR Code classes export { AccountQR } from './src/AccountQR'; export { ContactQR } from './src/ContactQR'; export { ObjectQR } from './src/ObjectQR'; export { TransactionQR } from './src/TransactionQR'; +export { CosignatureQR } from './src/CosignatureQR'; + +// factory export { QRCodeGenerator } from './src/QRCodeGenerator'; -export { QRService } from './src/QRService'; diff --git a/package-lock.json b/package-lock.json index ad1db38..2570896 100644 --- a/package-lock.json +++ b/package-lock.json @@ -139,6 +139,12 @@ "integrity": "sha512-1axi39YdtBI7z957vdqXI4Ac25e7YihYQtJa+Clnxg1zTJEaIRbndt71O3sP4GAMgiAm0pY26/b9BrY4MR/PMw==", "dev": true }, + "@types/node": { + "version": "12.0.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-12.0.2.tgz", + "integrity": "sha512-5tabW/i+9mhrfEOUcLDu2xBPsHJ+X5Orqy9FKpale3SjDA17j5AEpYq5vfy3oAeAHGcvANRCO3NV3d2D6q3NiA==", + "dev": true + }, "accepts": { "version": "1.3.7", "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", diff --git a/package.json b/package.json index 404d2c3..4d7e0f3 100644 --- a/package.json +++ b/package.json @@ -15,6 +15,7 @@ "devDependencies": { "@types/chai": "^4.1.3", "@types/mocha": "^5.2.0", + "@types/node": "^12.0.2", "chai": "^4.1.2", "coveralls": "^3.0.2", "mocha": "^5.2.0", diff --git a/src/AccountQR.ts b/src/AccountQR.ts index 078fecb..8d64384 100644 --- a/src/AccountQR.ts +++ b/src/AccountQR.ts @@ -17,7 +17,6 @@ import { Account, PublicAccount, NetworkType, - Wallet, Password, } from "nem2-sdk"; @@ -27,9 +26,11 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, - QRService + EncryptionService, + EncryptedPayload, + QRCodeDataSchema, + ExportAccountDataSchema } from '../index'; -import { throwError } from "rxjs"; export class AccountQR extends QRCode implements QRCodeInterface { /** @@ -49,7 +50,7 @@ export class AccountQR extends QRCode implements QRCodeInterface { * The password for encryption * @var {Password} */ - protected readonly password: Password, + public readonly password: Password, /** * The network type. * @var {NetworkType} @@ -64,30 +65,46 @@ export class AccountQR extends QRCode implements QRCodeInterface { } /** - * The `toJSON()` method should return the JSON - * representation of the QR Code content. + * Parse a JSON QR code content into a AccountQR + * object. * - * @return {string} + * @param json {string} + * @param password {Password} + * @return {AccountQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. */ - public toJSON(): string { + static fromJSON( + json: string, + password: Password + ): AccountQR { - if (this.password == null) { - throw Error('Password is missing'); - } - const qrService: QRService = new QRService(); - const encryption = qrService.AES_PBKF2_encryption(this.password, this.account.privateKey); + // create the QRCode object from JSON + return ExportAccountDataSchema.parse(json, password); + } - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': { - 'priv_key': encryption.encrypted, - 'salt': encryption.salt, - }, - }; + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @see https://en.wikipedia.org/wiki/QR_code#Storage + * @return {number} + */ + public getTypeNumber(): number { + // Type version for ContactQR is Version 10 + // This type of QR can hold up to 174 bytes of data. + return 10; + } - return JSON.stringify(jsonSchema); + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public getSchema(): QRCodeDataSchema { + return new ExportAccountDataSchema(); } } \ No newline at end of file diff --git a/src/ContactQR.ts b/src/ContactQR.ts index c2f6a8d..5d74696 100644 --- a/src/ContactQR.ts +++ b/src/ContactQR.ts @@ -17,6 +17,7 @@ import { Account, PublicAccount, NetworkType, + Address, } from "nem2-sdk"; // internal dependencies @@ -25,6 +26,8 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, + QRCodeDataSchema, + AddContactDataSchema } from '../index'; export class ContactQR extends QRCode implements QRCodeInterface { @@ -37,8 +40,13 @@ export class ContactQR extends QRCode implements QRCodeInterface { * @param chainId {string} */ constructor(/** + * The contact name. + * @var {string} + */ + public readonly name: string, + /** * The contact account. - * @var {Account|PublicAccount} + * @var {Account|PublicAccount|Address} */ public readonly account: Account | PublicAccount, /** @@ -55,23 +63,44 @@ export class ContactQR extends QRCode implements QRCodeInterface { } /** - * The `toJSON()` method should return the JSON - * representation of the QR Code content. + * Parse a JSON QR code content into a ContactQR + * object. * - * @return {string} + * @param json {string} + * @return {ContactQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. */ - public toJSON(): string { + static fromJSON( + json: string + ): ContactQR { - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': { - 'address': this.account - }, - }; + // create the QRCode object from JSON + return AddContactDataSchema.parse(json); + } + + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @see https://en.wikipedia.org/wiki/QR_code#Storage + * @return {number} + */ + public getTypeNumber(): number { + // Type version for ContactQR is Version 10 + // This type of QR can hold up to 174 bytes of data. + return 10; + } - return JSON.stringify(jsonSchema).trim(); + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public getSchema(): QRCodeDataSchema { + return new AddContactDataSchema(); } } \ No newline at end of file diff --git a/src/CosignatureQR.ts b/src/CosignatureQR.ts new file mode 100644 index 0000000..ee6bcf4 --- /dev/null +++ b/src/CosignatureQR.ts @@ -0,0 +1,101 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import { + Account, + PublicAccount, + NetworkType, +} from "nem2-sdk"; + +// internal dependencies +import { + QRCode, + QRCodeInterface, + QRCodeType, + QRCodeSettings, + QRCodeDataSchema, + RequestCosignatureDataSchema +} from '../index'; + +//XXX should it maybe extend TransactionQR to make use of version-40 ? +export class CosignatureQR extends QRCode implements QRCodeInterface { + /** + * Construct a Object QR Code out of the + * JSON object. + * + * @param object {Object} + * @param networkType {NetworkType} + * @param chainId {string} + */ + constructor(/** + * The hash of the transaction to co-sign + * @var {string} + */ + public readonly hash: string, + /** + * The network type. + * @var {NetworkType} + */ + public readonly networkType: NetworkType, + /** + * The chain Id. + * @var {string} + */ + public readonly chainId: string) { + super(QRCodeType.ExportObject, networkType, chainId); + } + + /** + * Parse a JSON QR code content into a CosignatureQR + * object. + * + * @param json {string} + * @return {CosignatureQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static fromJSON( + json: string + ): CosignatureQR { + + // create the QRCode object from JSON + return RequestCosignatureDataSchema.parse(json); + } + + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @see https://en.wikipedia.org/wiki/QR_code#Storage + * @return {number} + */ + public getTypeNumber(): number { + // Type version for ContactQR is Version 10 + // This type of QR can hold up to 174 bytes of data. + return 10; + } + + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public getSchema(): QRCodeDataSchema { + return new RequestCosignatureDataSchema(); + } +} \ No newline at end of file diff --git a/src/EncryptedPayload.ts b/src/EncryptedPayload.ts new file mode 100644 index 0000000..059b0f1 --- /dev/null +++ b/src/EncryptedPayload.ts @@ -0,0 +1,82 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +// internal dependencies +import {QRCodeType} from '../index'; + +/** + * Class `EncryptedPayload` describes an encrypted payload + * with salt and ciphertext properties. + * + * @since 0.3.0 + */ +export class EncryptedPayload { + + constructor(/** + * The payload ciphertext. + * The first X bytes represent the IV. + * + * @var {string} + */ + public readonly ciphertext: string, + /** + * The payload salt. + * + * @var {string} + */ + public readonly salt: string) {} + + /** + * Parse a JSON representation of an encrypted + * payload into a `EncryptedPayload` instance. + * + * The provided JSON must contain fields 'ciphertext' + * and 'salt'. + * + * @param {string} json + * @return {EncryptedPayload} + */ + public static fromJSON( + json: string + ): EncryptedPayload { + + if (! json.length) { + throw new Error('JSON argument cannot be empty.'); + } + + // validate JSON + let jsonObject: any; + try { + jsonObject = JSON.parse(json); + } + catch (e) { + // Invalid JSON provided, forward error + throw new Error(e); + } + + // validate obligatory fields + if (!jsonObject.hasOwnProperty('ciphertext')) { + throw new Error("Missing mandatory field 'ciphertext'."); + } + + if (!jsonObject.hasOwnProperty('salt')) { + throw new Error("Missing mandatory field 'salt'."); + } + + return new EncryptedPayload(jsonObject.ciphertext, jsonObject.salt); + } + + +} diff --git a/src/ObjectQR.ts b/src/ObjectQR.ts index 83a3d86..5688b61 100644 --- a/src/ObjectQR.ts +++ b/src/ObjectQR.ts @@ -25,6 +25,8 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, + QRCodeDataSchema, + ExportObjectDataSchema } from '../index'; export class ObjectQR extends QRCode implements QRCodeInterface { @@ -55,21 +57,44 @@ export class ObjectQR extends QRCode implements QRCodeInterface { } /** - * The `toJSON()` method should return the JSON - * representation of the QR Code content. + * Parse a JSON QR code content into a ObjectQR + * object. * - * @return {string} + * @param json {string} + * @return {ObjectQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. */ - public toJSON(): string { + static fromJSON( + json: string + ): ObjectQR { - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': this.object, - }; + // create the QRCode object from JSON + return ExportObjectDataSchema.parse(json); + } - return JSON.stringify(jsonSchema); + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @see https://en.wikipedia.org/wiki/QR_code#Storage + * @return {number} + */ + public getTypeNumber(): number { + // Type version for ContactQR is Version 10 + // This type of QR can hold up to 174 bytes of data. + return 10; + } + + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public getSchema(): QRCodeDataSchema { + return new ExportObjectDataSchema(); } } \ No newline at end of file diff --git a/src/QRCode.ts b/src/QRCode.ts index 64f432e..875e185 100644 --- a/src/QRCode.ts +++ b/src/QRCode.ts @@ -31,6 +31,7 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, + QRCodeDataSchema, } from "../index"; export abstract class QRCode implements QRCodeInterface { @@ -66,14 +67,40 @@ export abstract class QRCode implements QRCodeInterface { } /// region Abstract Methods + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public abstract getSchema(): QRCodeDataSchema; + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @return {number} + */ + public abstract getTypeNumber(): number; + /// end-region Abstract Methods + /** * The `toJSON()` method should return the JSON * representation of the QR Code content. * * @return {string} */ - public abstract toJSON(): string; - /// end-region Abstract Methods + public toJSON(): string { + + // get the QR Code Data Schema + const schema = this.getSchema(); + + // create the JSON object for this QR Code + const json = schema.toObject(this); + + // format to JSON + return JSON.stringify(json); + } /** * The `build()` method should return the QRCode @@ -85,7 +112,7 @@ export abstract class QRCode implements QRCodeInterface { // prepare QR generation const qr = new QRCodeImpl(); - qr.setTypeNumber(QRCodeSettings.VERSION_NUMBER); + qr.setTypeNumber(this.getTypeNumber()); qr.setErrorCorrectLevel(QRCodeSettings.CORRECTION_LEVEL); // get JSON representation diff --git a/src/QRCodeDataSchema.ts b/src/QRCodeDataSchema.ts new file mode 100644 index 0000000..00407d3 --- /dev/null +++ b/src/QRCodeDataSchema.ts @@ -0,0 +1,67 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +// internal dependencies +import {QRCode} from '../index'; + +/** + * Class `QRCodeDataSchema` describes a QR Code's data + * schema. The schema defines obligatory fields and their + * format. + * + * @since 0.3.0 + */ +export abstract class QRCodeDataSchema { + + /** + * The AccountQR QR Code version + * + * @var {number} + */ + public readonly VERSION = 3; + + constructor() {} + + /// region Abstract Methods + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public abstract getData(qr: QRCode): any; + /// end-region Abstract Methods + + /** + * The `toObject()` method returns a JSON object + * with required fields. + * + * @return {any} + */ + public toObject(qr: QRCode): any { + + // read data from child-classes + const data = this.getData(qr); + + return { + "v": this.VERSION, + "type": qr.type, + "network_id": qr.networkType, + "chain_id": qr.chainId, + "data": data + }; + } +} diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index 60e8874..6ba066c 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -31,7 +31,8 @@ import { ContactQR, ObjectQR, TransactionQR, - QRService, + CosignatureQR, + QRCode } from '../index'; /** @@ -89,11 +90,12 @@ export class QRCodeGenerator { * @param chainId {string} */ public static createContact( + name: string, account: Account | PublicAccount, networkType: NetworkType = NetworkType.MIJIN_TEST, chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' ): ContactQR { - return new ContactQR(account, networkType, chainId); + return new ContactQR(name, account, networkType, chainId); } public static createExportAccount( @@ -106,46 +108,66 @@ export class QRCodeGenerator { } /** - * Read JSON Content from QRcode. + * Parse a JSON QR code content into a sub-class + * of QRCode. + * * @param json {string} + * @return {QRCode} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. */ - static fromJSON(json:string, password?: Password) :any { + static fromJSON( + json: string, + password: Password | undefined = undefined + ): QRCode { - if (json == null || json == '') { - throw Error('QR json object is missing'); + if (! json.length) { + throw new Error('JSON argument cannot be empty.'); } - const jsonObj = JSON.parse(json || ''); + const jsonObj = JSON.parse(json); + if (!jsonObj.type) { + throw new Error('Missing mandatory field with name "type".'); + } - switch(jsonObj.type) { - case QRCodeType.AddContact: { - return new ContactQR(jsonObj.data.address, jsonObj.network_id, jsonObj.chainId) - } - case QRCodeType.ExportAccount: { - if (password == null){ - throw Error('Password are required'); - } + // We will use the `fromJSON` static implementation + // of specialized QRCode classes (child classes). + // An error will be thrown if the QRCodeType is not + // recognized or invalid. - const qrService: QRService = new QRService(); - const privatekey: string = qrService.AES_PBKF2_decryption(password,jsonObj); + switch (jsonObj.type) { - const account = Account.createFromPrivateKey(privatekey, - NetworkType.MIJIN_TEST); + // create a ContactQR from JSON + case QRCodeType.AddContact: + return ContactQR.fromJSON(json); - return new AccountQR(account, password, jsonObj.network_id, jsonObj.chainId) - } - case QRCodeType.RequestTransaction: { - let txMapping: Transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); + // create an AccountQR from JSON + case QRCodeType.ExportAccount: - return new TransactionQR(txMapping, jsonObj.network_id, jsonObj.chainId) - } - case QRCodeType.RequestCosignature: { - // Todo: In progress; - break; - } - case QRCodeType.ExportObject: { - return new ObjectQR(jsonObj.data.object, jsonObj.network_id, jsonObj.chainId); + // password obligatory for encryption + if (! password) { + throw new Error('Missing password to decrypt AccountQR QR code.'); } - } + + return AccountQR.fromJSON(json, password); + + // create a ObjectQR from JSON + case QRCodeType.ExportObject: + return ObjectQR.fromJSON(json); + + // create a CosignatureQR from JSON + case QRCodeType.RequestCosignature: + return CosignatureQR.fromJSON(json); + + // create a TransactionQR from JSON + case QRCodeType.RequestTransaction: + return TransactionQR.fromJSON(json); + + default: + break; + } + + throw new Error("Unrecognized QR Code 'type': '" + jsonObj.type + "'."); } } diff --git a/src/QRCodeSettings.ts b/src/QRCodeSettings.ts index a206a0a..fb1f92a 100644 --- a/src/QRCodeSettings.ts +++ b/src/QRCodeSettings.ts @@ -36,19 +36,11 @@ export class QRCodeSettings { public static CORRECTION_LEVEL = ErrorCorrectLevel.L; /** - * The QR Code version number. - * - * With `40-L` configuration, the QR Code can contain - * up to `2953` bytes. As defined in the following link, - * this is the maximum storage capacity for our types - * or QR Codes: - * - * https://en.wikipedia.org/wiki/QR_code#Design + * The NEM network QR Code version * - * @see https://en.wikipedia.org/wiki/QR_code#Design - * @var {string} + * @var {number} */ - public static VERSION_NUMBER = 40; + public static VERSION = 3; /** * The QR Code cell size in pixels. diff --git a/src/TransactionQR.ts b/src/TransactionQR.ts index 3d2fea8..5ca1d49 100644 --- a/src/TransactionQR.ts +++ b/src/TransactionQR.ts @@ -25,6 +25,8 @@ import { QRCodeInterface, QRCodeType, QRCodeSettings, + QRCodeDataSchema, + RequestTransactionDataSchema } from '../index'; export class TransactionQR extends QRCode implements QRCodeInterface { @@ -55,26 +57,44 @@ export class TransactionQR extends QRCode implements QRCodeInterface { } /** - * The `toJSON()` method should return the JSON - * representation of the QR Code content. + * Parse a JSON QR code content into a TransactionQR + * object. * - * @return {string} + * @param json {string} + * @return {TransactionQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. */ - public toJSON(): string { + static fromJSON( + json: string + ): TransactionQR { - // Serialize the transaction object data. - const txSerialized = this.transaction.serialize(); + // create the QRCode object from JSON + return RequestTransactionDataSchema.parse(json); + } - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': { - 'payload': txSerialized - } - }; + /** + * The `getTypeNumber()` method should return the + * version number for QR codes of the underlying class. + * + * @see https://en.wikipedia.org/wiki/QR_code#Storage + * @return {number} + */ + public getTypeNumber(): number { + // Type version for ContactQR is Version 40 + // This type of QR can hold up to 1264 bytes of data. + return 40; + } - return JSON.stringify(jsonSchema).trim(); + /** + * The `getSchema()` method should return an instance + * of a sub-class of QRCodeDataSchema which describes + * the QR Code data. + * + * @return {QRCodeDataSchema} + */ + public getSchema(): QRCodeDataSchema { + return new RequestTransactionDataSchema(); } } \ No newline at end of file diff --git a/src/schemas/AddContactDataSchema.ts b/src/schemas/AddContactDataSchema.ts new file mode 100644 index 0000000..8e7e7db --- /dev/null +++ b/src/schemas/AddContactDataSchema.ts @@ -0,0 +1,82 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import { + Address, + Account, + PublicAccount, +} from "nem2-sdk"; + +// internal dependencies +import { + QRCodeDataSchema, + QRCode, + QRCodeType, + ContactQR +} from '../../index'; + +/** + * Class `AddContactDataSchema` describes a contact + * add QR code data schema. + * + * @since 0.3.0 + */ +export class AddContactDataSchema extends QRCodeDataSchema { + + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public getData(qr: ContactQR): any { + return { + "name": qr.name, + "publicKey": qr.account.publicKey.toString(), + }; + } + + /** + * Parse a JSON QR code content into a ContactQR + * object. + * + * @param json {string} + * @return {ContactQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static parse( + json: string + ): ContactQR { + if (! json.length) { + throw Error('JSON argument cannot be empty.'); + } + + const jsonObj = JSON.parse(json); + if (!jsonObj.type || jsonObj.type !== QRCodeType.AddContact) { + throw Error('Invalid type field value for ContactQR.'); + } + + // read contact data + const name = jsonObj.data.name; + const network = jsonObj.network_id; + const account = PublicAccount.createFromPublicKey(jsonObj.data.publicKey, network); + const chainId = jsonObj.chain_id; + + return new ContactQR(name, account, network, chainId); + } +} diff --git a/src/schemas/ExportAccountDataSchema.ts b/src/schemas/ExportAccountDataSchema.ts new file mode 100644 index 0000000..6413034 --- /dev/null +++ b/src/schemas/ExportAccountDataSchema.ts @@ -0,0 +1,93 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import { + Account, + PublicAccount, + NetworkType, + Password, +} from "nem2-sdk"; + +// internal dependencies +import { + QRCodeDataSchema, + QRCode, + QRCodeType, + AccountQR, + EncryptionService, + EncryptedPayload, +} from '../../index'; + +/** + * Class `ExportAccountDataSchema` describes an export + * account QR code data schema. + * + * @since 0.3.0 + */ +export class ExportAccountDataSchema extends QRCodeDataSchema { + + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public getData(qr: AccountQR): any { + + // we will store a password encrypted copy of the private key + const encrypted = EncryptionService.encrypt(qr.account.privateKey, qr.password); + + return { + "ciphertext": encrypted.ciphertext, + "salt": encrypted.salt, + }; + } + + /** + * Parse a JSON QR code content into a AccountQR + * object. + * + * @param json {string} + * @param password {Password} + * @return {AccountQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static parse( + json: string, + password: Password + ): AccountQR { + if (! json.length) { + throw Error('JSON argument cannot be empty.'); + } + + const jsonObj = JSON.parse(json); + if (!jsonObj.type || jsonObj.type !== QRCodeType.ExportAccount) { + throw Error('Invalid type field value for AccountQR.'); + } + + // decrypt private key + const payload = new EncryptedPayload(jsonObj.data.ciphertext, jsonObj.data.salt); + const privKey = EncryptionService.decrypt(payload, password); + const network = jsonObj.network_id; + const chainId = jsonObj.chain_id; + + // create account + const account = Account.createFromPrivateKey(privKey, network); + return new AccountQR(account, password, network, chainId); + } +} diff --git a/src/schemas/ExportObjectDataSchema.ts b/src/schemas/ExportObjectDataSchema.ts new file mode 100644 index 0000000..a853ac1 --- /dev/null +++ b/src/schemas/ExportObjectDataSchema.ts @@ -0,0 +1,73 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +// internal dependencies +import { + QRCodeDataSchema, + QRCode, + QRCodeType, + ObjectQR +} from '../../index'; + +/** + * Class `ExportObjectDataSchema` describes an export + * object QR code data schema. + * + * @since 0.3.0 + */ +export class ExportObjectDataSchema extends QRCodeDataSchema { + + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public getData(qr: ObjectQR): any { + + return qr.object; + } + + /** + * Parse a JSON QR code content into a ObjectQR + * object. + * + * @param json {string} + * @return {ObjectQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static parse( + json: string + ): ObjectQR { + if (! json.length) { + throw Error('JSON argument cannot be empty.'); + } + + const jsonObj = JSON.parse(json); + if (!jsonObj.type || jsonObj.type !== QRCodeType.ExportObject) { + throw Error('Invalid type field value for ObjectQR.'); + } + + // read contact data + const obj = jsonObj.data; + const network = jsonObj.network_id; + const chainId = jsonObj.chain_id; + + return new ObjectQR(obj, network, chainId); + } +} diff --git a/src/schemas/RequestCosignatureDataSchema.ts b/src/schemas/RequestCosignatureDataSchema.ts new file mode 100644 index 0000000..057339f --- /dev/null +++ b/src/schemas/RequestCosignatureDataSchema.ts @@ -0,0 +1,74 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +// internal dependencies +import { + QRCodeDataSchema, + QRCode, + QRCodeType, + CosignatureQR +} from '../../index'; + +/** + * Class `RequestCosignatureDataSchema` describes a transaction + * request QR code data schema. + * + * @since 0.3.0 + */ +export class RequestCosignatureDataSchema extends QRCodeDataSchema { + + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public getData(qr: CosignatureQR): any { + return { + "hash": qr.hash + }; + } + + /** + * Parse a JSON QR code content into a CosignatureQR + * object. + * + * @param json {string} + * @return {CosignatureQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static parse( + json: string + ): CosignatureQR { + if (! json.length) { + throw Error('JSON argument cannot be empty.'); + } + + const jsonObj = JSON.parse(json); + if (!jsonObj.type || jsonObj.type !== QRCodeType.RequestCosignature) { + throw Error('Invalid type field value for CosignatureQR.'); + } + + // read contact data + const hash = jsonObj.data.hash; + const network = jsonObj.network_id; + const chainId = jsonObj.chain_id; + + return new CosignatureQR(hash, network, chainId); + } +} diff --git a/src/schemas/RequestTransactionDataSchema.ts b/src/schemas/RequestTransactionDataSchema.ts new file mode 100644 index 0000000..37b33f7 --- /dev/null +++ b/src/schemas/RequestTransactionDataSchema.ts @@ -0,0 +1,84 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import { + NetworkType, + TransactionMapping, + Transaction, +} from "nem2-sdk"; + +// internal dependencies +import { + QRCodeDataSchema, + QRCode, + QRCodeType, + TransactionQR +} from '../../index'; + +/** + * Class `RequestTransactionDataSchema` describes a transaction + * request QR code data schema. + * + * @since 0.3.0 + */ +export class RequestTransactionDataSchema extends QRCodeDataSchema { + + /** + * The `getData()` method returns an object + * that will be stored in the `data` field of + * the underlying QR Code JSON content. + * + * @return {any} + */ + public getData(qr: TransactionQR): any { + + // serialize the transaction object data. + const payload = qr.transaction.serialize(); + + return { + "payload": payload + }; + } + + /** + * Parse a JSON QR code content into a TransactionQR + * object. + * + * @param json {string} + * @return {TransactionQR} + * @throws {Error} On empty `json` given. + * @throws {Error} On missing `type` field value. + * @throws {Error} On unrecognized QR code `type` field value. + */ + static parse( + json: string + ): TransactionQR { + if (! json.length) { + throw Error('JSON argument cannot be empty.'); + } + + const jsonObj = JSON.parse(json); + if (!jsonObj.type || jsonObj.type !== QRCodeType.RequestTransaction) { + throw Error('Invalid type field value for TransactionQR.'); + } + + // read contact data + const transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); + const network = jsonObj.network_id; + const chainId = jsonObj.chain_id; + + return new TransactionQR(transaction, network, chainId); + } +} diff --git a/src/services/EncryptionService.ts b/src/services/EncryptionService.ts new file mode 100644 index 0000000..048f4fb --- /dev/null +++ b/src/services/EncryptionService.ts @@ -0,0 +1,127 @@ +/* + * Copyright (c) 2019 NEM Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License "); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import { + convert, + nacl_catapult, + sha3Hasher +} from 'nem2-library'; +import { + Password, +} from "nem2-sdk"; +import * as CryptoJS from "crypto-js"; + +// internal dependencies +import { + EncryptedPayload +} from '../../index'; + +/** + * Class `EncryptionService` describes a high level service + * for encryption/decryption of data. + * + * Implemented algorithms for encryption/decryption include: + * - AES with PBKDF2 (Password-Based Key Derivation Function) + * + * @since 0.3.0 + */ +export class EncryptionService { + + /** + * The `encrypt` method will encrypt given `data` raw string + * with given `password` password. + * + * First we generate a random salt of 32 bytes, then we iterate + * 2000 times with PBKDF2 and encrypt with AES. + * + * @param password {Password} + * @param data {string} + */ + public static encrypt( + data: string, + password: Password + ): EncryptedPayload { + + // create random salt (32 bytes) + const salt = CryptoJS.lib.WordArray.random(32); + + // derive key of 8 bytes with 2000 iterations of PBKDF2 + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 8, + iterations: 2000, + }); + + // create encryption input vector of 16 bytes (iv) + const iv = CryptoJS.lib.WordArray.random(16); + + // format IV for crypto-js encryption + const encIv = { + iv: iv + }; + + // encrypt with AES + const dataBin = CryptoJS.lib.WordArray.create(data); + const encrypted = CryptoJS.AES.encrypt(dataBin, key, encIv); + + // create our `EncryptedPayload` + const ciphertext = iv.toString() + encrypted.toString(); + const used_salt = salt.toString(); + + return new EncryptedPayload(ciphertext, used_salt); + } + + /** + * AES_PBKF2_decryption will decrypt privateKey with provided password + * @param password + * @param json + */ + public static decrypt( + payloadOrJson: EncryptedPayload | string, + password: Password + ): string { + + let payload: EncryptedPayload; + + // parse input if necessary + if (payloadOrJson instanceof EncryptedPayload) { + payload = payloadOrJson; + } + else { + payload = EncryptedPayload.fromJSON(payloadOrJson); + } + + // read payload + const salt = CryptoJS.enc.Hex.parse(payload.salt); + const priv = payload.ciphertext; + + // read encryption configuration + const iv: string = CryptoJS.enc.Hex.parse(priv.substring(0, 32)); + const cipher: string = priv.substring(32, 96); + + const encIv = { + iv: iv + }; + + // re-generate key (PBKDF2) + const key = CryptoJS.PBKDF2(password.value, salt, { + keySize: 8, + iterations: 2000, + }); + + // decrypt and return + const decrypted = CryptoJS.AES.decrypt(cipher, key, encIv); + return decrypted.toString(); + } +} \ No newline at end of file diff --git a/test/ContactQR.spec.ts b/test/ContactQR.spec.ts index c74f0c3..90b11ac 100644 --- a/test/ContactQR.spec.ts +++ b/test/ContactQR.spec.ts @@ -39,13 +39,14 @@ describe('ContactQR -->', () => { it('include mandatory NIP-7 QR Code base fields', () => { // Arrange: + const name = 'test-contact-1'; const account = PublicAccount.createFromPublicKey( 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', NetworkType.TEST_NET ); // Act: - const addContact = new ContactQR(account, NetworkType.TEST_NET, ''); + const addContact = new ContactQR(name, account, NetworkType.TEST_NET, ''); const actualJSON = addContact.toJSON(); const actualObject = JSON.parse(actualJSON); diff --git a/test/QRCode.spec.ts b/test/QRCode.spec.ts index cca45b2..f6868d6 100644 --- a/test/QRCode.spec.ts +++ b/test/QRCode.spec.ts @@ -36,6 +36,8 @@ import { QRCode, QRCodeType, QRCodeSettings, + QRCodeDataSchema, + ExportObjectDataSchema, } from "../index"; /// region Mock for QRCode specialization @@ -48,15 +50,12 @@ class FakeQR extends QRCode implements QRCodeInterface { super(QRCodeType.ExportObject, networkType, chainId); } - public toJSON(): string { - const jsonSchema = { - 'v': 3, - 'type': this.type, - 'network_id': this.networkType, - 'chain_id': this.chainId, - 'data': this.object, - }; - return JSON.stringify(jsonSchema); + public getSchema(): QRCodeDataSchema { + return new ExportObjectDataSchema(); + } + + public getTypeNumber(): number { + return 10; } } /// end-region Mock for QRCode specialization @@ -101,14 +100,14 @@ describe('QRCode -->', () => { it('set correct settings for QR Code generation', () => { // Arrange: const object = {"test": "test"}; - const modulesCount = QRCodeSettings.VERSION_NUMBER * 4 + 17; + const modulesCount = 10 * 4 + 17; // Act: const fakeQR = new FakeQR(object, NetworkType.TEST_NET, 'no-chain-id'); const implQR = fakeQR.build(); // Assert: - expect(implQR.getTypeNumber()).to.be.equal(QRCodeSettings.VERSION_NUMBER); + expect(implQR.getTypeNumber()).to.be.equal(10); expect(implQR.getErrorCorrectLevel()).to.be.equal(QRCodeSettings.CORRECTION_LEVEL); expect(implQR.getModuleCount()).to.be.equal(modulesCount); }); diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index 0677869..391b276 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -29,7 +29,13 @@ import { } from 'nem2-sdk'; // internal dependencies -import { QRCodeGenerator, AccountQR, TransactionQR, QRCodeType, ContactQR } from "../index"; +import { + QRCodeGenerator, + AccountQR, + TransactionQR, + QRCodeType, + ContactQR +} from "../index"; // vectors data import { @@ -86,13 +92,14 @@ describe('QRCodeGenerator -->', () => { it('generate correct Base64 representation for TransferTransaction', () => { // Arrange: + const name = 'test-contact-1'; const account = PublicAccount.createFromPublicKey( 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', NetworkType.MIJIN_TEST ); // Act: - const createContact = QRCodeGenerator.createContact(account); + const createContact = QRCodeGenerator.createContact(name, account); const actualBase64 = createContact.toBase64(); // Assert: @@ -142,7 +149,7 @@ describe('QRCodeGenerator -->', () => { const txJSON = requestTx.toJSON(); // Act: - const transactionObj: TransactionQR = QRCodeGenerator.fromJSON(txJSON); + const transactionObj: TransactionQR = QRCodeGenerator.fromJSON(txJSON) as TransactionQR; // Assert: expect(transactionObj).to.not.be.equal(''); @@ -152,17 +159,22 @@ describe('QRCodeGenerator -->', () => { it('Read data From ContactQR', () => { // Arrange: + const name = 'test-contact-1'; const account = PublicAccount.createFromPublicKey( 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', NetworkType.MIJIN_TEST ); - const createContact = QRCodeGenerator.createContact(account,NetworkType.MIJIN_TEST); + const createContact = QRCodeGenerator.createContact( + name, + account, + NetworkType.MIJIN_TEST + ); const contactJSON = createContact.toJSON(); // Act: - const contactObj: ContactQR = QRCodeGenerator.fromJSON(contactJSON); + const contactObj: ContactQR = QRCodeGenerator.fromJSON(contactJSON) as ContactQR; // Assert: expect(contactObj).to.not.be.equal(''); @@ -182,7 +194,7 @@ describe('QRCodeGenerator -->', () => { const actualObj = exportAccount.toJSON(); // Act: - const accountObj: AccountQR = QRCodeGenerator.fromJSON(actualObj,password); + const accountObj: AccountQR = QRCodeGenerator.fromJSON(actualObj,password) as AccountQR; // Assert: expect(accountObj).to.not.be.equal(''); From 23a1294398151d6a2785a8cc4467d91cf9bc7373 Mon Sep 17 00:00:00 2001 From: Greg S Date: Thu, 23 May 2019 14:24:18 +0200 Subject: [PATCH 10/13] nemtech/NIP#21 : add specialized schema unit tests, fixed CosignatureQR, TransactionQR --- index.ts | 2 +- src/CosignatureQR.ts | 33 +++--- src/TransactionQR.ts | 10 +- src/schemas/AddContactDataSchema.ts | 4 + src/schemas/ExportAccountDataSchema.ts | 4 + src/schemas/ExportObjectDataSchema.ts | 4 + src/schemas/RequestCosignatureDataSchema.ts | 33 +++--- src/schemas/RequestTransactionDataSchema.ts | 4 + test/AccountQR.spec.ts | 22 +++- test/ContactQR.spec.ts | 18 +++ test/CosignatureQR.spec.ts | 115 ++++++++++++++++++++ test/TransactionQR.spec.ts | 22 ++++ 12 files changed, 233 insertions(+), 38 deletions(-) create mode 100644 test/CosignatureQR.spec.ts diff --git a/index.ts b/index.ts index 80eb989..d94a13a 100644 --- a/index.ts +++ b/index.ts @@ -29,8 +29,8 @@ export { QRCodeDataSchema } from './src/QRCodeDataSchema'; export { AddContactDataSchema } from './src/schemas/AddContactDataSchema'; export { ExportAccountDataSchema } from './src/schemas/ExportAccountDataSchema'; export { ExportObjectDataSchema } from './src/schemas/ExportObjectDataSchema'; -export { RequestCosignatureDataSchema } from './src/schemas/RequestCosignatureDataSchema'; export { RequestTransactionDataSchema } from './src/schemas/RequestTransactionDataSchema'; +export { RequestCosignatureDataSchema } from './src/schemas/RequestCosignatureDataSchema'; // specialized QR Code classes export { AccountQR } from './src/AccountQR'; diff --git a/src/CosignatureQR.ts b/src/CosignatureQR.ts index ee6bcf4..2d75de5 100644 --- a/src/CosignatureQR.ts +++ b/src/CosignatureQR.ts @@ -14,9 +14,10 @@ *limitations under the License. */ import { - Account, - PublicAccount, NetworkType, + TransactionMapping, + Transaction, + AggregateTransaction, } from "nem2-sdk"; // internal dependencies @@ -26,24 +27,24 @@ import { QRCodeType, QRCodeSettings, QRCodeDataSchema, - RequestCosignatureDataSchema + RequestCosignatureDataSchema, + TransactionQR, } from '../index'; -//XXX should it maybe extend TransactionQR to make use of version-40 ? -export class CosignatureQR extends QRCode implements QRCodeInterface { +export class CosignatureQR extends TransactionQR implements QRCodeInterface { /** - * Construct a Object QR Code out of the - * JSON object. - * - * @param object {Object} + * Construct a Transaction Request QR Code out of the + * nem2-sdk Transaction instance. + * + * @param transaction {Transaction} * @param networkType {NetworkType} * @param chainId {string} */ constructor(/** - * The hash of the transaction to co-sign - * @var {string} + * The transaction for the request. + * @var {AggregateTransaction} */ - public readonly hash: string, + public readonly transaction: AggregateTransaction, /** * The network type. * @var {NetworkType} @@ -54,7 +55,7 @@ export class CosignatureQR extends QRCode implements QRCodeInterface { * @var {string} */ public readonly chainId: string) { - super(QRCodeType.ExportObject, networkType, chainId); + super(transaction, networkType, chainId, QRCodeType.RequestCosignature); } /** @@ -83,9 +84,9 @@ export class CosignatureQR extends QRCode implements QRCodeInterface { * @return {number} */ public getTypeNumber(): number { - // Type version for ContactQR is Version 10 - // This type of QR can hold up to 174 bytes of data. - return 10; + // Type version for ContactQR is Version 40 + // This type of QR can hold up to 1264 bytes of data. + return 40; } /** diff --git a/src/TransactionQR.ts b/src/TransactionQR.ts index 5ca1d49..d5efba1 100644 --- a/src/TransactionQR.ts +++ b/src/TransactionQR.ts @@ -52,8 +52,14 @@ export class TransactionQR extends QRCode implements QRCodeInterface { * The chain Id. * @var {string} */ - public readonly chainId: string) { - super(QRCodeType.RequestTransaction, networkType, chainId); + public readonly chainId: string, + /** + * The QR Code Type + * + * @var {QRCodeType} + */ + public readonly type: QRCodeType = QRCodeType.RequestTransaction) { + super(type, networkType, chainId); } /** diff --git a/src/schemas/AddContactDataSchema.ts b/src/schemas/AddContactDataSchema.ts index 8e7e7db..0123237 100644 --- a/src/schemas/AddContactDataSchema.ts +++ b/src/schemas/AddContactDataSchema.ts @@ -35,6 +35,10 @@ import { */ export class AddContactDataSchema extends QRCodeDataSchema { + constructor() { + super(); + } + /** * The `getData()` method returns an object * that will be stored in the `data` field of diff --git a/src/schemas/ExportAccountDataSchema.ts b/src/schemas/ExportAccountDataSchema.ts index 6413034..a82bdc4 100644 --- a/src/schemas/ExportAccountDataSchema.ts +++ b/src/schemas/ExportAccountDataSchema.ts @@ -38,6 +38,10 @@ import { */ export class ExportAccountDataSchema extends QRCodeDataSchema { + constructor() { + super(); + } + /** * The `getData()` method returns an object * that will be stored in the `data` field of diff --git a/src/schemas/ExportObjectDataSchema.ts b/src/schemas/ExportObjectDataSchema.ts index a853ac1..84bd700 100644 --- a/src/schemas/ExportObjectDataSchema.ts +++ b/src/schemas/ExportObjectDataSchema.ts @@ -29,6 +29,10 @@ import { */ export class ExportObjectDataSchema extends QRCodeDataSchema { + constructor() { + super(); + } + /** * The `getData()` method returns an object * that will be stored in the `data` field of diff --git a/src/schemas/RequestCosignatureDataSchema.ts b/src/schemas/RequestCosignatureDataSchema.ts index 057339f..39f89bb 100644 --- a/src/schemas/RequestCosignatureDataSchema.ts +++ b/src/schemas/RequestCosignatureDataSchema.ts @@ -13,33 +13,32 @@ * See the License for the specific language governing permissions and *limitations under the License. */ +import { + NetworkType, + TransactionMapping, + Transaction, + AggregateTransaction, +} from "nem2-sdk"; + // internal dependencies import { QRCodeDataSchema, QRCode, QRCodeType, - CosignatureQR + CosignatureQR, + RequestTransactionDataSchema, } from '../../index'; /** * Class `RequestCosignatureDataSchema` describes a transaction - * request QR code data schema. + * cosignature request QR code data schema. * * @since 0.3.0 */ -export class RequestCosignatureDataSchema extends QRCodeDataSchema { +export class RequestCosignatureDataSchema extends RequestTransactionDataSchema { - /** - * The `getData()` method returns an object - * that will be stored in the `data` field of - * the underlying QR Code JSON content. - * - * @return {any} - */ - public getData(qr: CosignatureQR): any { - return { - "hash": qr.hash - }; + constructor() { + super(); } /** @@ -60,15 +59,15 @@ export class RequestCosignatureDataSchema extends QRCodeDataSchema { } const jsonObj = JSON.parse(json); - if (!jsonObj.type || jsonObj.type !== QRCodeType.RequestCosignature) { + if (!jsonObj.type || jsonObj.type !== QRCodeType.RequestTransaction) { throw Error('Invalid type field value for CosignatureQR.'); } // read contact data - const hash = jsonObj.data.hash; + const transaction = TransactionMapping.createFromPayload(jsonObj.data.payload); const network = jsonObj.network_id; const chainId = jsonObj.chain_id; - return new CosignatureQR(hash, network, chainId); + return new CosignatureQR(transaction as AggregateTransaction, network, chainId); } } diff --git a/src/schemas/RequestTransactionDataSchema.ts b/src/schemas/RequestTransactionDataSchema.ts index 37b33f7..17b0ca8 100644 --- a/src/schemas/RequestTransactionDataSchema.ts +++ b/src/schemas/RequestTransactionDataSchema.ts @@ -35,6 +35,10 @@ import { */ export class RequestTransactionDataSchema extends QRCodeDataSchema { + constructor() { + super(); + } + /** * The `getData()` method returns an object * that will be stored in the `data` field of diff --git a/test/AccountQR.spec.ts b/test/AccountQR.spec.ts index 17b5787..55ec23e 100644 --- a/test/AccountQR.spec.ts +++ b/test/AccountQR.spec.ts @@ -39,7 +39,7 @@ describe('AccountQR -->', () => { describe('toJSON() should', () => { - it.only('include mandatory NIP-7 QR Code base fields', () => { + it('include mandatory NIP-7 QR Code base fields', () => { // Arrange: const account = Account.createFromPrivateKey( 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', @@ -52,13 +52,31 @@ describe('AccountQR -->', () => { const actualJSON = exportAccount.toJSON(); const actualObject = JSON.parse(actualJSON); - // // // Assert: + // Assert: expect(actualObject).to.have.property('v'); expect(actualObject).to.have.property('type'); expect(actualObject).to.have.property('network_id'); expect(actualObject).to.have.property('chain_id'); expect(actualObject).to.have.property('data'); }); + + it('include specialized schema fields', () => { + // Arrange: + const account = Account.createFromPrivateKey( + 'F97AE23C2A28ECEDE6F8D6C447C0A10B55C92DDE9316CCD36C3177B073906978', + NetworkType.MIJIN_TEST + ); + const password = new Password('password'); + + // Act: + const exportAccount = new AccountQR(account, password, NetworkType.MIJIN_TEST, 'no-chain-id'); + const actualJSON = exportAccount.toJSON(); + const actualObject = JSON.parse(actualJSON); + + // Assert: + expect(actualObject.data).to.have.property('ciphertext'); + expect(actualObject.data).to.have.property('salt'); + }); }); }); diff --git a/test/ContactQR.spec.ts b/test/ContactQR.spec.ts index 90b11ac..2fccdc5 100644 --- a/test/ContactQR.spec.ts +++ b/test/ContactQR.spec.ts @@ -57,6 +57,24 @@ describe('ContactQR -->', () => { expect(actualObject).to.have.property('chain_id'); expect(actualObject).to.have.property('data'); }); + + it('include specialized schema fields', () => { + // Arrange: + const name = 'test-contact-1'; + const account = PublicAccount.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.TEST_NET + ); + + // Act: + const addContact = new ContactQR(name, account, NetworkType.TEST_NET, ''); + const actualJSON = addContact.toJSON(); + const actualObject = JSON.parse(actualJSON); + + // Assert: + expect(actualObject.data).to.have.property('name'); + expect(actualObject.data).to.have.property('publicKey'); + }); }); }); diff --git a/test/CosignatureQR.spec.ts b/test/CosignatureQR.spec.ts new file mode 100644 index 0000000..65f20c4 --- /dev/null +++ b/test/CosignatureQR.spec.ts @@ -0,0 +1,115 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import {expect} from "chai"; +import { + AggregateTransaction, + TransferTransaction, + Deadline, + Address, + Mosaic, + NamespaceId, + UInt64, + PlainMessage, + NetworkType, + PublicAccount, +} from 'nem2-sdk'; +import { + QRCode as QRCodeImpl, + QR8BitByte, + ErrorCorrectLevel, +} from 'qrcode-generator-ts'; + +// internal dependencies +import { + QRCodeInterface, + QRCode, + QRCodeType, + QRCodeSettings, + CosignatureQR, +} from "../index"; + +describe('CosignatureQR -->', () => { + + describe('toJSON() should', () => { + + it('include mandatory NIP-7 QR Code base fields', () => { + // Arrange: + const account = PublicAccount.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ); + const transfer = TransferTransaction.create( + Deadline.create(), + Address.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ), + [new Mosaic(new NamespaceId('cat.currency'), UInt64.fromUint(10000000))], + PlainMessage.create('Welcome to NEM!'), + NetworkType.MIJIN_TEST + ); + const bonded = AggregateTransaction.createBonded( + Deadline.create(), + [transfer.toAggregate(account)], + NetworkType.MIJIN_TEST + ); + + // Act: + const requestTx = new CosignatureQR(bonded, NetworkType.TEST_NET, ''); + const actualJSON = requestTx.toJSON(); + const actualObject = JSON.parse(actualJSON); + + // Assert: + expect(actualObject).to.have.property('v'); + expect(actualObject).to.have.property('type'); + expect(actualObject).to.have.property('network_id'); + expect(actualObject).to.have.property('chain_id'); + expect(actualObject).to.have.property('data'); + }); + + it('include specialized schema fields', () => { + // Arrange: + const account = PublicAccount.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ); + const transfer = TransferTransaction.create( + Deadline.create(), + Address.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ), + [new Mosaic(new NamespaceId('cat.currency'), UInt64.fromUint(10000000))], + PlainMessage.create('Welcome to NEM!'), + NetworkType.MIJIN_TEST + ); + const bonded = AggregateTransaction.createBonded( + Deadline.create(), + [transfer.toAggregate(account)], + NetworkType.MIJIN_TEST + ); + + // Act: + const requestTx = new CosignatureQR(bonded, NetworkType.TEST_NET, ''); + const actualJSON = requestTx.toJSON(); + const actualObject = JSON.parse(actualJSON); + + // Assert: + expect(actualObject.data).to.have.property('payload'); + }); + }); + +}); diff --git a/test/TransactionQR.spec.ts b/test/TransactionQR.spec.ts index 281f71c..57a7b02 100644 --- a/test/TransactionQR.spec.ts +++ b/test/TransactionQR.spec.ts @@ -68,6 +68,28 @@ describe('TransactionQR -->', () => { expect(actualObject).to.have.property('chain_id'); expect(actualObject).to.have.property('data'); }); + + it('include specialized schema fields', () => { + // Arrange: + const transfer = TransferTransaction.create( + Deadline.create(), + Address.createFromPublicKey( + 'C5C55181284607954E56CD46DE85F4F3EF4CC713CC2B95000FA741998558D268', + NetworkType.MIJIN_TEST + ), + [new Mosaic(new NamespaceId('cat.currency'), UInt64.fromUint(10000000))], + PlainMessage.create('Welcome to NEM!'), + NetworkType.MIJIN_TEST + ); + + // Act: + const requestTx = new TransactionQR(transfer, NetworkType.TEST_NET, ''); + const actualJSON = requestTx.toJSON(); + const actualObject = JSON.parse(actualJSON); + + // Assert: + expect(actualObject.data).to.have.property('payload'); + }); }); }); From ca6eb07d4e42c32dd3c58c80051393290b693cbf Mon Sep 17 00:00:00 2001 From: Greg S Date: Mon, 27 May 2019 20:53:57 +0200 Subject: [PATCH 11/13] nemtech/NIP#21 : WIP on unit tests, fixed AccountQR, WIP on EncryptionService --- index.ts | 8 +-- src/AccountQR.ts | 4 +- src/ContactQR.ts | 2 +- src/CosignatureQR.ts | 2 +- src/EncryptedPayload.ts | 1 - src/ObjectQR.ts | 2 +- src/QRCode.ts | 2 +- src/QRCodeGenerator.ts | 68 ++++++++++++++--------- src/QRCodeSettings.ts | 7 --- src/QRService.ts | 77 -------------------------- src/TransactionQR.ts | 2 +- src/schemas/ExportAccountDataSchema.ts | 4 ++ test/AccountQR.spec.ts | 12 +--- test/ContactQR.spec.ts | 11 +--- test/CosignatureQR.spec.ts | 6 +- test/EncryptedPayload.spec.ts | 75 +++++++++++++++++++++++++ test/EncryptionService.spec.ts | 76 +++++++++++++++++++++++++ test/QRCode.spec.ts | 9 +-- test/QRCodeGenerator.spec.ts | 53 ++++++++++-------- test/TransactionQR.spec.ts | 11 +--- test/vectors/OBJECT.ts | 16 ------ test/vectors/index.ts | 16 ------ 22 files changed, 244 insertions(+), 220 deletions(-) delete mode 100644 src/QRService.ts create mode 100644 test/EncryptedPayload.spec.ts create mode 100644 test/EncryptionService.spec.ts delete mode 100644 test/vectors/OBJECT.ts delete mode 100644 test/vectors/index.ts diff --git a/index.ts b/index.ts index d94a13a..0767a12 100644 --- a/index.ts +++ b/index.ts @@ -20,10 +20,6 @@ export { QRCodeSettings } from './src/QRCodeSettings'; export { QRCodeInterface } from './src/QRCodeInterface'; export { QRCode } from './src/QRCode'; -// encryption -export { EncryptedPayload } from './src/EncryptedPayload'; -export { EncryptionService } from './src/services/EncryptionService'; - // QR Code data schemas export { QRCodeDataSchema } from './src/QRCodeDataSchema'; export { AddContactDataSchema } from './src/schemas/AddContactDataSchema'; @@ -32,6 +28,10 @@ export { ExportObjectDataSchema } from './src/schemas/ExportObjectDataSchema'; export { RequestTransactionDataSchema } from './src/schemas/RequestTransactionDataSchema'; export { RequestCosignatureDataSchema } from './src/schemas/RequestCosignatureDataSchema'; +// encryption +export { EncryptedPayload } from './src/EncryptedPayload'; +export { EncryptionService } from './src/services/EncryptionService'; + // specialized QR Code classes export { AccountQR } from './src/AccountQR'; export { ContactQR } from './src/ContactQR'; diff --git a/src/AccountQR.ts b/src/AccountQR.ts index 8d64384..7da0dfd 100644 --- a/src/AccountQR.ts +++ b/src/AccountQR.ts @@ -94,7 +94,7 @@ export class AccountQR extends QRCode implements QRCodeInterface { public getTypeNumber(): number { // Type version for ContactQR is Version 10 // This type of QR can hold up to 174 bytes of data. - return 10; + return 20; } /** @@ -107,4 +107,4 @@ export class AccountQR extends QRCode implements QRCodeInterface { public getSchema(): QRCodeDataSchema { return new ExportAccountDataSchema(); } -} \ No newline at end of file +} diff --git a/src/ContactQR.ts b/src/ContactQR.ts index 5d74696..6e14500 100644 --- a/src/ContactQR.ts +++ b/src/ContactQR.ts @@ -103,4 +103,4 @@ export class ContactQR extends QRCode implements QRCodeInterface { public getSchema(): QRCodeDataSchema { return new AddContactDataSchema(); } -} \ No newline at end of file +} diff --git a/src/CosignatureQR.ts b/src/CosignatureQR.ts index 2d75de5..463444b 100644 --- a/src/CosignatureQR.ts +++ b/src/CosignatureQR.ts @@ -99,4 +99,4 @@ export class CosignatureQR extends TransactionQR implements QRCodeInterface { public getSchema(): QRCodeDataSchema { return new RequestCosignatureDataSchema(); } -} \ No newline at end of file +} diff --git a/src/EncryptedPayload.ts b/src/EncryptedPayload.ts index 059b0f1..d822bfc 100644 --- a/src/EncryptedPayload.ts +++ b/src/EncryptedPayload.ts @@ -78,5 +78,4 @@ export class EncryptedPayload { return new EncryptedPayload(jsonObject.ciphertext, jsonObject.salt); } - } diff --git a/src/ObjectQR.ts b/src/ObjectQR.ts index 5688b61..64ce12a 100644 --- a/src/ObjectQR.ts +++ b/src/ObjectQR.ts @@ -97,4 +97,4 @@ export class ObjectQR extends QRCode implements QRCodeInterface { public getSchema(): QRCodeDataSchema { return new ExportObjectDataSchema(); } -} \ No newline at end of file +} diff --git a/src/QRCode.ts b/src/QRCode.ts index 875e185..060c607 100644 --- a/src/QRCode.ts +++ b/src/QRCode.ts @@ -140,4 +140,4 @@ export abstract class QRCode implements QRCodeInterface { QRCodeSettings.MARGIN_PIXEL ); } -} \ No newline at end of file +} diff --git a/src/QRCodeGenerator.ts b/src/QRCodeGenerator.ts index 6ba066c..0d68c5e 100644 --- a/src/QRCodeGenerator.ts +++ b/src/QRCodeGenerator.ts @@ -21,18 +21,17 @@ import { PublicAccount, Password } from "nem2-sdk"; -import * as CryptoJS from "crypto-js"; // internal dependencies import { QRCodeInterface, QRCodeType, + QRCode, AccountQR, ContactQR, ObjectQR, TransactionQR, - CosignatureQR, - QRCode + CosignatureQR } from '../index'; /** @@ -53,6 +52,7 @@ export class QRCodeGenerator { /** * Create a JSON object QR Code from a JSON object. * + * @see {ObjectQR} * @param object {Object} * @param networkType {NetworkType} * @param chainId {string} @@ -66,47 +66,58 @@ export class QRCodeGenerator { } /** - * Create a Transaction Request QR Code from a Transaction - * instance. + * Create a Contact QR Code from a contact name + * and account. * + * @see {ContactQR} * @param transaction {Transaction} * @param networkType {NetworkType} * @param chainId {string} */ - public static createTransactionRequest( - transaction: Transaction, + public static createAddContact( + name: string, + account: Account | PublicAccount, networkType: NetworkType = NetworkType.MIJIN_TEST, chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' - ): TransactionQR { - return new TransactionQR(transaction, networkType, chainId); + ): ContactQR { + return new ContactQR(name, account, networkType, chainId); } /** * Create a Transaction Request QR Code from a Transaction * instance. * + * @see {AccountQR} * @param transaction {Transaction} * @param networkType {NetworkType} * @param chainId {string} */ - public static createContact( - name: string, - account: Account | PublicAccount, - networkType: NetworkType = NetworkType.MIJIN_TEST, - chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' - ): ContactQR { - return new ContactQR(name, account, networkType, chainId); - } - public static createExportAccount( account: Account, password: Password, networkType: NetworkType = NetworkType.MIJIN_TEST, - chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31', + chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' ): AccountQR { return new AccountQR(account, password, networkType, chainId); } + /** + * Create a Transaction Request QR Code from a Transaction + * instance. + * + * @see {TransactionQR} + * @param transaction {Transaction} + * @param networkType {NetworkType} + * @param chainId {string} + */ + public static createTransactionRequest( + transaction: Transaction, + networkType: NetworkType = NetworkType.MIJIN_TEST, + chainId: string = 'E2A9F95E129283EF47B92A62FD748DBA4D32AA718AE6F8AC99C105CFA9F27A31' + ): TransactionQR { + return new TransactionQR(transaction, networkType, chainId); + } + /** * Parse a JSON QR code content into a sub-class * of QRCode. @@ -126,9 +137,16 @@ export class QRCodeGenerator { throw new Error('JSON argument cannot be empty.'); } - const jsonObj = JSON.parse(json); - if (!jsonObj.type) { - throw new Error('Missing mandatory field with name "type".'); + let jsonObject: any; + try { + jsonObject = JSON.parse(json); + if (!jsonObject.type) { + throw new Error('Missing mandatory field with name "type".'); + } + } + catch(e) { + // Invalid JSON provided, forward error + throw new Error(e); } // We will use the `fromJSON` static implementation @@ -136,7 +154,7 @@ export class QRCodeGenerator { // An error will be thrown if the QRCodeType is not // recognized or invalid. - switch (jsonObj.type) { + switch (jsonObject.type) { // create a ContactQR from JSON case QRCodeType.AddContact: @@ -151,7 +169,7 @@ export class QRCodeGenerator { } return AccountQR.fromJSON(json, password); - + // create a ObjectQR from JSON case QRCodeType.ExportObject: return ObjectQR.fromJSON(json); @@ -168,6 +186,6 @@ export class QRCodeGenerator { break; } - throw new Error("Unrecognized QR Code 'type': '" + jsonObj.type + "'."); + throw new Error("Unrecognized QR Code 'type': '" + jsonObject.type + "'."); } } diff --git a/src/QRCodeSettings.ts b/src/QRCodeSettings.ts index fb1f92a..595bb9c 100644 --- a/src/QRCodeSettings.ts +++ b/src/QRCodeSettings.ts @@ -35,13 +35,6 @@ export class QRCodeSettings { */ public static CORRECTION_LEVEL = ErrorCorrectLevel.L; - /** - * The NEM network QR Code version - * - * @var {number} - */ - public static VERSION = 3; - /** * The QR Code cell size in pixels. * diff --git a/src/QRService.ts b/src/QRService.ts deleted file mode 100644 index ea3bb58..0000000 --- a/src/QRService.ts +++ /dev/null @@ -1,77 +0,0 @@ -/* - * Copyright (c) 2019 NEM Foundation - * - * Licensed under the Apache License, Version 2.0 (the "License "); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -import { - Password, -} from "nem2-sdk"; -import {convert,nacl_catapult} from 'nem2-library'; -import * as CryptoJS from "crypto-js"; - -export class QRService { - - /** - * AES_PBKF2_encryption will encrypt privateKey with provided password. - * @param password {Password} - * @param privateKey {strinf} - */ - public AES_PBKF2_encryption(password: Password, privateKey: string): any { - const salt = CryptoJS.lib.WordArray.random(256 / 8); - const key = CryptoJS.PBKDF2(password.value, salt, { - keySize: 256 / 32, - iterations: 2000, - }); - - const hex = convert.uint8ToHex(nacl_catapult.randomBytes(16)); - - const encIv = { - iv: CryptoJS.enc.Hex.parse(hex), - }; - - const encrypted = CryptoJS.AES.encrypt(CryptoJS.enc.Hex.parse(privateKey), key, encIv); - - return { - encrypted: hex + encrypted.toString(), - salt: salt.toString(), - }; - } - - /** - * AES_PBKF2_decryption will decrypt privateKey with provided password - * @param password - * @param json - */ - public AES_PBKF2_decryption(password: Password, json: any): string { - const encryptedData = json; - const salt = CryptoJS.enc.Hex.parse(encryptedData.data.salt); - const iv = CryptoJS.enc.Hex.parse(encryptedData.data.priv_key.substring(0, 32)); - const encrypted: string = encryptedData.data.priv_key.substring(32, 96); - - //generate key - const key = CryptoJS.PBKDF2(password.value, salt, { - keySize: 256 / 32, - iterations: 2000, - }); - - let encIv = { - iv: iv - }; - - let decrypt = CryptoJS.enc.Hex.stringify(CryptoJS.AES.decrypt(encrypted, key, encIv)); - - if (decrypt === "" || (decrypt.length != 64 && decrypt.length != 66)) throw new Error("invalid password"); - return decrypt; - } -} \ No newline at end of file diff --git a/src/TransactionQR.ts b/src/TransactionQR.ts index d5efba1..232a8d6 100644 --- a/src/TransactionQR.ts +++ b/src/TransactionQR.ts @@ -103,4 +103,4 @@ export class TransactionQR extends QRCode implements QRCodeInterface { public getSchema(): QRCodeDataSchema { return new RequestTransactionDataSchema(); } -} \ No newline at end of file +} diff --git a/src/schemas/ExportAccountDataSchema.ts b/src/schemas/ExportAccountDataSchema.ts index a82bdc4..c03e4e7 100644 --- a/src/schemas/ExportAccountDataSchema.ts +++ b/src/schemas/ExportAccountDataSchema.ts @@ -84,12 +84,16 @@ export class ExportAccountDataSchema extends QRCodeDataSchema { throw Error('Invalid type field value for AccountQR.'); } + console.log("JSON: ", jsonObj); + // decrypt private key const payload = new EncryptedPayload(jsonObj.data.ciphertext, jsonObj.data.salt); const privKey = EncryptionService.decrypt(payload, password); const network = jsonObj.network_id; const chainId = jsonObj.chain_id; + console.log("data: ", privKey); + // create account const account = Account.createFromPrivateKey(privKey, network); return new AccountQR(account, password, network, chainId); diff --git a/test/AccountQR.spec.ts b/test/AccountQR.spec.ts index 55ec23e..bc581ef 100644 --- a/test/AccountQR.spec.ts +++ b/test/AccountQR.spec.ts @@ -19,21 +19,11 @@ import { NetworkType, Password, } from 'nem2-sdk'; -import { - QRCode as QRCodeImpl, - QR8BitByte, - ErrorCorrectLevel, -} from 'qrcode-generator-ts'; // internal dependencies import { - QRCodeInterface, - QRCode, - QRCodeType, - QRCodeSettings, - ContactQR, + AccountQR, } from "../index"; -import { AccountQR } from "../src/AccountQR"; describe('AccountQR -->', () => { diff --git a/test/ContactQR.spec.ts b/test/ContactQR.spec.ts index 2fccdc5..cc8bd41 100644 --- a/test/ContactQR.spec.ts +++ b/test/ContactQR.spec.ts @@ -18,18 +18,9 @@ import { PublicAccount, NetworkType, } from 'nem2-sdk'; -import { - QRCode as QRCodeImpl, - QR8BitByte, - ErrorCorrectLevel, -} from 'qrcode-generator-ts'; // internal dependencies -import { - QRCodeInterface, - QRCode, - QRCodeType, - QRCodeSettings, +import { ContactQR, } from "../index"; diff --git a/test/CosignatureQR.spec.ts b/test/CosignatureQR.spec.ts index 65f20c4..db4b8c9 100644 --- a/test/CosignatureQR.spec.ts +++ b/test/CosignatureQR.spec.ts @@ -33,11 +33,7 @@ import { } from 'qrcode-generator-ts'; // internal dependencies -import { - QRCodeInterface, - QRCode, - QRCodeType, - QRCodeSettings, +import { CosignatureQR, } from "../index"; diff --git a/test/EncryptedPayload.spec.ts b/test/EncryptedPayload.spec.ts new file mode 100644 index 0000000..356cd30 --- /dev/null +++ b/test/EncryptedPayload.spec.ts @@ -0,0 +1,75 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import {expect} from "chai"; + +// internal dependencies +import { + EncryptedPayload, +} from "../index"; + +describe('EncryptedPayload -->', () => { + + describe('fromJSON() should', () => { + + it('throw on empty JSON', () => { + // Arrange: + const json = ''; + + // Act + Assert + expect((function () { + const payload = EncryptedPayload.fromJSON(json); + })).to.throw('JSON argument cannot be empty.'); + }); + + it('throw on missing ciphertext property', () => { + // Arrange: + const json = '{"salt": "00"}'; + + // Act + Assert + expect((function () { + const payload = EncryptedPayload.fromJSON(json); + })).to.throw('Missing mandatory field \'ciphertext\'.'); + }); + + it('throw on missing salt property', () => { + // Arrange: + const json = '{"ciphertext": "00"}'; + + // Act + Assert + expect((function () { + const payload = EncryptedPayload.fromJSON(json); + })).to.throw('Missing mandatory field \'salt\'.'); + }); + + it('create complete object', () => { + // Arrange: + const json = { + ciphertext: "zyFIAqnq8fihaJFqgH9gVKGT1Aa8dbxXqrcWb//Ckv7R/DJDgdXOY8ejc6KNURPGujULpv0fQnN87AQFldmCgkGYq0CBSHwhOhyCvEBK18g=", + salt: "12345678901234567890123456789012" + }; + + // Act + const payload = EncryptedPayload.fromJSON(JSON.stringify(json)); + + // Assert + expect(payload.ciphertext).to.not.be.undefined; + expect(payload.ciphertext).to.be.equal(json.ciphertext); + expect(payload.salt).to.not.be.undefined; + expect(payload.salt).to.be.equal(json.salt); + }); + }); + +}); diff --git a/test/EncryptionService.spec.ts b/test/EncryptionService.spec.ts new file mode 100644 index 0000000..95fce30 --- /dev/null +++ b/test/EncryptionService.spec.ts @@ -0,0 +1,76 @@ +/** + * Copyright 2019 NEM + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + *limitations under the License. + */ +import {expect} from "chai"; +import {Password} from 'nem2-sdk'; + +// internal dependencies +import { + EncryptionService, + EncryptedPayload, +} from "../index"; + +describe('EncryptionService -->', () => { + + describe('encrypt() should', () => { + + it('should create encrypted payload with salt', () => { + // Arrange: + const data = 'this will be encrypted.'; + const pass = 'password'; + + // Act + const encrypted = EncryptionService.encrypt(data, new Password(pass)); + + // Asset + expect(encrypted.ciphertext).to.not.be.undefined; + expect(encrypted.salt).to.not.be.undefined; + expect(encrypted.salt).to.have.lengthOf(64); + }); + + it('should create correctly sized ciphertext and salt', () => { + // Arrange: + const data = 'this will be encrypted.'; + const pass = 'password'; + + // Act + const encrypted = EncryptionService.encrypt(data, new Password(pass)); + + // Asset + expect(encrypted.ciphertext).to.have.lengthOf(160); + expect(encrypted.salt).to.have.lengthOf(64); + }); + + it('should always create different ciphertext with salt', () => { + // Arrange: + const data = 'this will be encrypted.'; + const pass = 'password'; + + // Act + const encrypted_1 = EncryptionService.encrypt(data, new Password(pass)); + const encrypted_2 = EncryptionService.encrypt(data, new Password(pass)); + const encrypted_3 = EncryptionService.encrypt(data, new Password(pass)); + + // Asset + expect(encrypted_1).to.not.be.equal(encrypted_2); + expect(encrypted_1).to.not.be.equal(encrypted_3); + expect(encrypted_2).to.not.be.equal(encrypted_3); + expect(encrypted_1.salt).to.have.lengthOf(64); + expect(encrypted_2.salt).to.have.lengthOf(64); + expect(encrypted_3.salt).to.have.lengthOf(64); + }); + }); + +}); diff --git a/test/QRCode.spec.ts b/test/QRCode.spec.ts index f6868d6..85d3892 100644 --- a/test/QRCode.spec.ts +++ b/test/QRCode.spec.ts @@ -15,13 +15,6 @@ */ import {expect} from "chai"; import { - TransferTransaction, - Deadline, - Address, - Mosaic, - NamespaceId, - UInt64, - PlainMessage, NetworkType, } from 'nem2-sdk'; import { @@ -31,7 +24,7 @@ import { } from 'qrcode-generator-ts'; // internal dependencies -import { +import { QRCodeInterface, QRCode, QRCodeType, diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index 391b276..cd9b1a6 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -30,33 +30,39 @@ import { // internal dependencies import { + QRCodeType, QRCodeGenerator, + ContactQR, AccountQR, TransactionQR, - QRCodeType, - ContactQR } from "../index"; -// vectors data -import { - ExpectedObjectBase64, -} from './vectors/index'; - describe('QRCodeGenerator -->', () => { describe('createExportObject() should', () => { - it('generate correct Base64 representation for {test: test}', () => { + + it('use default values for network_id and chain_id', () => { // Arrange: - const object = {"test": "test"}; - const qrcode = QRCodeGenerator.createExportObject(object, NetworkType.TEST_NET, 'no-chain-id'); + const object = {}; // Act: - const base64 = qrcode.toBase64(); + const objectQR = QRCodeGenerator.createExportObject(object); // Assert: - expect(base64).to.not.be.equal(''); - expect(base64.length).to.not.be.equal(0); - expect(base64).to.be.equal(ExpectedObjectBase64); + expect(objectQR.networkType).to.be.equal(NetworkType.MIJIN_TEST); + expect(objectQR.chainId).to.not.be.undefined; + expect(objectQR.chainId).to.have.lengthOf(64); + }); + + it('fill object property correctly with {test: test}', () => { + // Arrange: + const object = {test: "test"}; + + // Act: + const objectQR = QRCodeGenerator.createExportObject(object); + + // Assert: + expect(objectQR.object).to.deep.equal(object); }); }); @@ -88,7 +94,7 @@ describe('QRCodeGenerator -->', () => { }); }); - describe('createContact() should', ()=> { + describe('createAddContact() should', () => { it('generate correct Base64 representation for TransferTransaction', () => { // Arrange: @@ -99,7 +105,7 @@ describe('QRCodeGenerator -->', () => { ); // Act: - const createContact = QRCodeGenerator.createContact(name, account); + const createContact = QRCodeGenerator.createAddContact(name, account); const actualBase64 = createContact.toBase64(); // Assert: @@ -109,7 +115,7 @@ describe('QRCodeGenerator -->', () => { }); }); - describe('createExportAccount() should', ()=> { + describe('createExportAccount() should', () => { it('generate correct Base64 representation for ExportAccount', () => { // Arrange: @@ -120,7 +126,7 @@ describe('QRCodeGenerator -->', () => { const password = new Password('password'); // Act: - const exportAccount = QRCodeGenerator.createExportAccount(account,password); + const exportAccount = QRCodeGenerator.createExportAccount(account, password); const actualBase64 = exportAccount.toBase64(); // Assert: @@ -165,7 +171,7 @@ describe('QRCodeGenerator -->', () => { NetworkType.MIJIN_TEST ); - const createContact = QRCodeGenerator.createContact( + const createContact = QRCodeGenerator.createAddContact( name, account, NetworkType.MIJIN_TEST @@ -181,7 +187,7 @@ describe('QRCodeGenerator -->', () => { expect(contactObj.account.address).to.deep.equal(account.address); expect(contactObj.type).to.deep.equal(QRCodeType.AddContact); }); - +/* it('Read data From AccountQR', () => { // Arrange: const account = Account.createFromPrivateKey( @@ -190,18 +196,19 @@ describe('QRCodeGenerator -->', () => { ); const password = new Password('password'); - const exportAccount = QRCodeGenerator.createExportAccount(account,password); + const exportAccount = QRCodeGenerator.createExportAccount(account, password); const actualObj = exportAccount.toJSON(); // Act: - const accountObj: AccountQR = QRCodeGenerator.fromJSON(actualObj,password) as AccountQR; + const accountObj: AccountQR = QRCodeGenerator.fromJSON(actualObj, password) as AccountQR; // Assert: expect(accountObj).to.not.be.equal(''); expect(accountObj.account).to.deep.equal(account); expect(accountObj.type).to.deep.equal(QRCodeType.ExportAccount); }); - +*/ it('Read data From ObjectQR', () => {}); }); + }); diff --git a/test/TransactionQR.spec.ts b/test/TransactionQR.spec.ts index 57a7b02..accf6b4 100644 --- a/test/TransactionQR.spec.ts +++ b/test/TransactionQR.spec.ts @@ -24,18 +24,9 @@ import { PlainMessage, NetworkType, } from 'nem2-sdk'; -import { - QRCode as QRCodeImpl, - QR8BitByte, - ErrorCorrectLevel, -} from 'qrcode-generator-ts'; // internal dependencies -import { - QRCodeInterface, - QRCode, - QRCodeType, - QRCodeSettings, +import { TransactionQR, } from "../index"; diff --git a/test/vectors/OBJECT.ts b/test/vectors/OBJECT.ts deleted file mode 100644 index a93418b..0000000 --- a/test/vectors/OBJECT.ts +++ /dev/null @@ -1,16 +0,0 @@ -/** - * Copyright 2019 NEM - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - *limitations under the License. - */ -export const ExpectedObjectBase64 = ""; diff --git a/test/vectors/index.ts b/test/vectors/index.ts deleted file mode 100644 index bb94d19..0000000 --- a/test/vectors/index.ts +++ /dev/null @@ -1,16 +0,0 @@ -/** - * Copyright 2019 NEM - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - *limitations under the License. - */ -export { ExpectedObjectBase64 } from './OBJECT'; From 991fc7df96aa2f143ca6ca04df1a9a561a0e7ea8 Mon Sep 17 00:00:00 2001 From: Greg S Date: Mon, 27 May 2019 21:27:08 +0200 Subject: [PATCH 12/13] nemtech/NIP#21 : fixed EncryptionService encrypt() and decrypt(), added unit tests for encryption --- src/services/EncryptionService.ts | 33 +++++++++++++++---------------- test/EncryptionService.spec.ts | 24 ++++++++++++++++++---- 2 files changed, 36 insertions(+), 21 deletions(-) diff --git a/src/services/EncryptionService.ts b/src/services/EncryptionService.ts index 048f4fb..0612b0a 100644 --- a/src/services/EncryptionService.ts +++ b/src/services/EncryptionService.ts @@ -66,18 +66,16 @@ export class EncryptionService { // create encryption input vector of 16 bytes (iv) const iv = CryptoJS.lib.WordArray.random(16); - // format IV for crypto-js encryption - const encIv = { - iv: iv - }; - // encrypt with AES - const dataBin = CryptoJS.lib.WordArray.create(data); - const encrypted = CryptoJS.AES.encrypt(dataBin, key, encIv); + const encrypted = CryptoJS.AES.encrypt(data, key, { + iv: iv, + padding: CryptoJS.pad.Pkcs7, + mode: CryptoJS.mode.CBC + }); - // create our `EncryptedPayload` + // create our `EncryptedPayload` (16 bytes iv as hex || cipher text) const ciphertext = iv.toString() + encrypted.toString(); - const used_salt = salt.toString(); + const used_salt = CryptoJS.enc.Hex.stringify(salt); return new EncryptedPayload(ciphertext, used_salt); } @@ -107,12 +105,8 @@ export class EncryptionService { const priv = payload.ciphertext; // read encryption configuration - const iv: string = CryptoJS.enc.Hex.parse(priv.substring(0, 32)); - const cipher: string = priv.substring(32, 96); - - const encIv = { - iv: iv - }; + const iv: string = CryptoJS.enc.Hex.parse(priv.substr(0, 32)); + const cipher: string = priv.substr(32); // re-generate key (PBKDF2) const key = CryptoJS.PBKDF2(password.value, salt, { @@ -121,7 +115,12 @@ export class EncryptionService { }); // decrypt and return - const decrypted = CryptoJS.AES.decrypt(cipher, key, encIv); - return decrypted.toString(); + const decrypted = CryptoJS.AES.decrypt(cipher, key, { + iv: iv, + padding: CryptoJS.pad.Pkcs7, + mode: CryptoJS.mode.CBC + }); + + return decrypted.toString(CryptoJS.enc.Utf8); } } \ No newline at end of file diff --git a/test/EncryptionService.spec.ts b/test/EncryptionService.spec.ts index 95fce30..615e19e 100644 --- a/test/EncryptionService.spec.ts +++ b/test/EncryptionService.spec.ts @@ -34,7 +34,7 @@ describe('EncryptionService -->', () => { // Act const encrypted = EncryptionService.encrypt(data, new Password(pass)); - // Asset + // Assert expect(encrypted.ciphertext).to.not.be.undefined; expect(encrypted.salt).to.not.be.undefined; expect(encrypted.salt).to.have.lengthOf(64); @@ -48,8 +48,8 @@ describe('EncryptionService -->', () => { // Act const encrypted = EncryptionService.encrypt(data, new Password(pass)); - // Asset - expect(encrypted.ciphertext).to.have.lengthOf(160); + // Assert + expect(encrypted.ciphertext).to.have.lengthOf(76); expect(encrypted.salt).to.have.lengthOf(64); }); @@ -63,7 +63,7 @@ describe('EncryptionService -->', () => { const encrypted_2 = EncryptionService.encrypt(data, new Password(pass)); const encrypted_3 = EncryptionService.encrypt(data, new Password(pass)); - // Asset + // Assert expect(encrypted_1).to.not.be.equal(encrypted_2); expect(encrypted_1).to.not.be.equal(encrypted_3); expect(encrypted_2).to.not.be.equal(encrypted_3); @@ -73,4 +73,20 @@ describe('EncryptionService -->', () => { }); }); + describe('decrypt() should', () => { + + it('should decrypt ciphertext correctly', () => { + // Arrange: + const data = 'this will be encrypted'; + const pass = 'password'; + + // Act + const encrypted = EncryptionService.encrypt(data, new Password(pass)); + const decrypted = EncryptionService.decrypt(encrypted, new Password(pass)); + + // Assert + expect(decrypted).to.be.equal(data); + }); + }); + }); From 70354f7536b731adad4edd694e5952772a07fc5e Mon Sep 17 00:00:00 2001 From: Greg S Date: Mon, 27 May 2019 21:28:23 +0200 Subject: [PATCH 13/13] nemtech/NIP#21 : fixed AccountQR and QRCodeGenerator.fromJSON for AccountQR --- CHANGELOG.md | 28 +++++++++++++++++++++----- README.md | 1 + src/schemas/ExportAccountDataSchema.ts | 4 ---- test/QRCodeGenerator.spec.ts | 4 ++-- 4 files changed, 26 insertions(+), 11 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cbb66a..9813198 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,14 +1,32 @@ # CHANGELOG -## v0.1.0 +## v0.3.0 -- added qr-library. -- generate QRcode by string. -- generate image base64 string by string. +- Added class `EncryptedPayload` +- Added class `EncryptionService` +- Added data schemas structure with `QRCodeDataSchema` +- Added data schema `AddContactDataSchema` +- Added data schema `ExportAccountDataSchema` +- Added data schema `RequestCosignatureDataSchema` child of Transaction data schema +- Added data schema `RequestTransactionDataSchema` child of Transaction data schema +- Unit tests for AccountQR, ContactQR, TransactionQR, CosignatureQR +- Modified QRCode.toJSON() logic to make use of `build()` method +- Fixed `AccountQR` generation of encrypted private keys for accounts +- Fixed `ContactQR` to also hold `name` information (optional) +- Fixed QR Code `TypeNumber`: ContactQR uses type 10, AccountQR type 20, TransactionQR type 40. +- Removed class `QRService` +- Removed encryption from `QRService` in `AccountQR` ## v0.2.0 + - added QRcode base class - added AccountQR Class - added ContactQR Class - added TransactionQR Class -- added ObjectQR Class \ No newline at end of file +- added ObjectQR Class + +## v0.1.0 + +- added qr-library. +- generate QRcode by string. +- generate image base64 string by string. diff --git a/README.md b/README.md index beda9b1..94fd806 100644 --- a/README.md +++ b/README.md @@ -66,6 +66,7 @@ The produced Base64 encoded payload can be used to display the QR Code. An examp Important versions listed below. Refer to the [Changelog](CHANGELOG.md) for a full history of the project. +- [0.3.0](CHANGELOG.md) - 2019-05-27 - [0.2.0](CHANGELOG.md) - 2019-05-01 - [0.1.0](CHANGELOG.md) - 2019-04-20 diff --git a/src/schemas/ExportAccountDataSchema.ts b/src/schemas/ExportAccountDataSchema.ts index c03e4e7..a82bdc4 100644 --- a/src/schemas/ExportAccountDataSchema.ts +++ b/src/schemas/ExportAccountDataSchema.ts @@ -84,16 +84,12 @@ export class ExportAccountDataSchema extends QRCodeDataSchema { throw Error('Invalid type field value for AccountQR.'); } - console.log("JSON: ", jsonObj); - // decrypt private key const payload = new EncryptedPayload(jsonObj.data.ciphertext, jsonObj.data.salt); const privKey = EncryptionService.decrypt(payload, password); const network = jsonObj.network_id; const chainId = jsonObj.chain_id; - console.log("data: ", privKey); - // create account const account = Account.createFromPrivateKey(privKey, network); return new AccountQR(account, password, network, chainId); diff --git a/test/QRCodeGenerator.spec.ts b/test/QRCodeGenerator.spec.ts index cd9b1a6..ba1c028 100644 --- a/test/QRCodeGenerator.spec.ts +++ b/test/QRCodeGenerator.spec.ts @@ -187,7 +187,7 @@ describe('QRCodeGenerator -->', () => { expect(contactObj.account.address).to.deep.equal(account.address); expect(contactObj.type).to.deep.equal(QRCodeType.AddContact); }); -/* + it('Read data From AccountQR', () => { // Arrange: const account = Account.createFromPrivateKey( @@ -207,7 +207,7 @@ describe('QRCodeGenerator -->', () => { expect(accountObj.account).to.deep.equal(account); expect(accountObj.type).to.deep.equal(QRCodeType.ExportAccount); }); -*/ + it('Read data From ObjectQR', () => {}); });