From 23d6a76b1b048117c1f81cb69d506c11fda9f8a9 Mon Sep 17 00:00:00 2001 From: aaron67 Date: Mon, 25 Mar 2024 09:40:27 +0800 Subject: [PATCH 1/8] Add contract --- src/contracts/oracleDemoBsv20.ts | 94 ++++++++++++++++++++++++++++++++ 1 file changed, 94 insertions(+) create mode 100644 src/contracts/oracleDemoBsv20.ts diff --git a/src/contracts/oracleDemoBsv20.ts b/src/contracts/oracleDemoBsv20.ts new file mode 100644 index 00000000..0474c304 --- /dev/null +++ b/src/contracts/oracleDemoBsv20.ts @@ -0,0 +1,94 @@ +import { + ByteString, + SmartContract, + Utils, + assert, + method, + prop, + reverseByteString, + slice, +} from 'scrypt-ts' +import { RabinPubKey, RabinSig, WitnessOnChainVerifier } from 'scrypt-ts-lib' + +type Msg = { + marker: bigint // 1 byte, api marker + timestamp: bigint // 4 bytes LE + network: bigint // 1 byte, 1 for mainnet, 0 for testnet + txid: ByteString // 32 bytes, txid + vout: bigint // 4 bytes LE, output index + bsv20: bigint // 1 byte, token type, 0 for NFT, 1 for BSV20 + amt: bigint // 8 bytes LE + id: ByteString // 66 bytes +} + +export class OracleDemoBsv20 extends SmartContract { + @prop() + oraclePubKey: RabinPubKey + + @prop() + inscriptionId: ByteString + @prop() + amt: bigint + + constructor( + oraclePubKey: RabinPubKey, + inscriptionId: ByteString, + amt: bigint + ) { + super(...arguments) + this.oraclePubKey = oraclePubKey + this.inscriptionId = inscriptionId + this.amt = amt + } + + @method() + static parseMsg(msg: ByteString): Msg { + return { + marker: Utils.fromLEUnsigned(slice(msg, 0n, 1n)), + timestamp: Utils.fromLEUnsigned(slice(msg, 1n, 5n)), + network: Utils.fromLEUnsigned(slice(msg, 5n, 6n)), + txid: slice(msg, 6n, 38n), + vout: Utils.fromLEUnsigned(slice(msg, 38n, 42n)), + bsv20: Utils.fromLEUnsigned(slice(msg, 42n, 43n)), + amt: Utils.fromLEUnsigned(slice(msg, 43n, 51n)), + id: slice(msg, 51n, 117n), + } + } + + @method() + public unlock(msg: ByteString, sig: RabinSig, tokenInputIndex: bigint) { + this.ctx + // retrieve token outpoint from prevouts + const txid = reverseByteString( + slice( + this.prevouts, + tokenInputIndex * 36n, + tokenInputIndex * 36n + 32n + ), + 32n + ) + const vout = Utils.fromLEUnsigned( + slice( + this.prevouts, + tokenInputIndex * 36n + 32n, + tokenInputIndex * 36n + 36n + ) + ) + // verify oracle signature + assert( + WitnessOnChainVerifier.verifySig(msg, sig, this.oraclePubKey), + 'Oracle sig verify failed.' + ) + // decode oracle data + const message = OracleDemoBsv20.parseMsg(msg) + // validate data + assert(message.marker == 4n, 'incorrect oracle message type') + assert(message.network == 0n, 'incorrect network') + assert(message.txid == txid, 'incorrect token txid') + assert(message.vout == vout, 'incorrect token vout') + assert(message.bsv20 == 1n, 'incorrect token type') + assert(message.amt >= this.amt, 'incorrect token amount') + assert(message.id == this.inscriptionId, 'incorrect inscription id') + // do other validations ... + } +} From 08b2d77c126aae553c5c5da2b05f0185a19a03d7 Mon Sep 17 00:00:00 2001 From: hh Date: Mon, 25 Mar 2024 10:23:12 +0800 Subject: [PATCH 2/8] remove ctx --- src/contracts/oracleDemoBsv20.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/contracts/oracleDemoBsv20.ts b/src/contracts/oracleDemoBsv20.ts index 0474c304..3e3bd9cd 100644 --- a/src/contracts/oracleDemoBsv20.ts +++ b/src/contracts/oracleDemoBsv20.ts @@ -57,7 +57,6 @@ export class OracleDemoBsv20 extends SmartContract { @method() public unlock(msg: ByteString, sig: RabinSig, tokenInputIndex: bigint) { - this.ctx // retrieve token outpoint from prevouts const txid = reverseByteString( slice( From 59570211d09ec43604c92150d450e68e1f598714 Mon Sep 17 00:00:00 2001 From: hh Date: Mon, 25 Mar 2024 17:52:10 +0800 Subject: [PATCH 3/8] pump scrypt-cli --- package-lock.json | 34 +++++++++++++++++----------------- package.json | 2 +- 2 files changed, 18 insertions(+), 18 deletions(-) diff --git a/package-lock.json b/package-lock.json index 57fbea42..89f48f72 100644 --- a/package-lock.json +++ b/package-lock.json @@ -33,7 +33,7 @@ "mocha": "^10.1.0", "prettier": "^2.8.2", "rimraf": "^3.0.2", - "scrypt-cli": "^0.1.68", + "scrypt-cli": "^0.1.72", "ts-node": "^10.9.1", "typescript": "^5.3.3" } @@ -4534,9 +4534,9 @@ ] }, "node_modules/scrypt-cli": { - "version": "0.1.68", - "resolved": "https://registry.npmjs.org/scrypt-cli/-/scrypt-cli-0.1.68.tgz", - "integrity": "sha512-+z8a3SlcRk3ghxVx/cdTW8Vsi3t25pbRTtKuoJu6GWduiFTDgQ+zobJfc71ep5PQUJy3AprNUZNrRdcY8b7PTA==", + "version": "0.1.72", + "resolved": "https://registry.npmjs.org/scrypt-cli/-/scrypt-cli-0.1.72.tgz", + "integrity": "sha512-jLfewoDSnmKWeoliQ7GjP9VeCbfkUjIaxBDalGabpzwb2BA4QVvVVbQIlHBz+bh2jK8NlK1/Gi7RjQJqPPpGsw==", "dev": true, "dependencies": { "axios": "^1.3.6", @@ -4552,8 +4552,8 @@ "json5": "^2.2.2", "lodash": "^4.17.21", "ora": "^5.4.1", - "scrypt-ts-transpiler": "^1.2.15", - "scryptlib": "^2.1.35", + "scrypt-ts-transpiler": "^1.2.20", + "scryptlib": "^2.1.41", "semver": "^7.3.8", "shelljs": "^0.8.5", "table": "^6.8.0", @@ -4761,9 +4761,9 @@ } }, "node_modules/scrypt-ts-transpiler": { - "version": "1.2.16", - "resolved": "https://registry.npmjs.org/scrypt-ts-transpiler/-/scrypt-ts-transpiler-1.2.16.tgz", - "integrity": "sha512-/0H/B1BT2M9lVkR8vcTHUSsHUb7I1aO7zba9eghM9TQhG5Th/oprQh64KysAxMlYJX+8FAi8upQRq8RJ27zoqw==", + "version": "1.2.20", + "resolved": "https://registry.npmjs.org/scrypt-ts-transpiler/-/scrypt-ts-transpiler-1.2.20.tgz", + "integrity": "sha512-fYS11B9W8NOzGoCeQ/ORFrmRPuZEo1NZEGt0HXpgivPsQB873q+p/bcBXCBrfWzN7cesfMYU9IoNYPHcNGcvlA==", "dev": true, "hasInstallScript": true, "dependencies": { @@ -9203,9 +9203,9 @@ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" }, "scrypt-cli": { - "version": "0.1.68", - "resolved": "https://registry.npmjs.org/scrypt-cli/-/scrypt-cli-0.1.68.tgz", - "integrity": "sha512-+z8a3SlcRk3ghxVx/cdTW8Vsi3t25pbRTtKuoJu6GWduiFTDgQ+zobJfc71ep5PQUJy3AprNUZNrRdcY8b7PTA==", + "version": "0.1.72", + "resolved": "https://registry.npmjs.org/scrypt-cli/-/scrypt-cli-0.1.72.tgz", + "integrity": "sha512-jLfewoDSnmKWeoliQ7GjP9VeCbfkUjIaxBDalGabpzwb2BA4QVvVVbQIlHBz+bh2jK8NlK1/Gi7RjQJqPPpGsw==", "dev": true, "requires": { "axios": "^1.3.6", @@ -9221,8 +9221,8 @@ "json5": "^2.2.2", "lodash": "^4.17.21", "ora": "^5.4.1", - "scrypt-ts-transpiler": "^1.2.15", - "scryptlib": "^2.1.35", + "scrypt-ts-transpiler": "^1.2.20", + "scryptlib": "^2.1.41", "semver": "^7.3.8", "shelljs": "^0.8.5", "table": "^6.8.0", @@ -9380,9 +9380,9 @@ } }, "scrypt-ts-transpiler": { - "version": "1.2.16", - "resolved": "https://registry.npmjs.org/scrypt-ts-transpiler/-/scrypt-ts-transpiler-1.2.16.tgz", - "integrity": "sha512-/0H/B1BT2M9lVkR8vcTHUSsHUb7I1aO7zba9eghM9TQhG5Th/oprQh64KysAxMlYJX+8FAi8upQRq8RJ27zoqw==", + "version": "1.2.20", + "resolved": "https://registry.npmjs.org/scrypt-ts-transpiler/-/scrypt-ts-transpiler-1.2.20.tgz", + "integrity": "sha512-fYS11B9W8NOzGoCeQ/ORFrmRPuZEo1NZEGt0HXpgivPsQB873q+p/bcBXCBrfWzN7cesfMYU9IoNYPHcNGcvlA==", "dev": true, "requires": { "@phenomnomnominal/tsquery": "^6.1.2", diff --git a/package.json b/package.json index bbe6646a..8872f8dc 100644 --- a/package.json +++ b/package.json @@ -52,7 +52,7 @@ "mocha": "^10.1.0", "prettier": "^2.8.2", "rimraf": "^3.0.2", - "scrypt-cli": "^0.1.68", + "scrypt-cli": "^0.1.72", "ts-node": "^10.9.1", "typescript": "^5.3.3" } From c76144792de11440defb26fb68ab382b074b13d1 Mon Sep 17 00:00:00 2001 From: hh Date: Mon, 25 Mar 2024 17:52:27 +0800 Subject: [PATCH 4/8] update watch task --- .vscode/tasks.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.vscode/tasks.json b/.vscode/tasks.json index 150a6bcb..3d6091a6 100644 --- a/.vscode/tasks.json +++ b/.vscode/tasks.json @@ -5,7 +5,7 @@ "label": "Build", "detail": "Build the project which includes `scrypt-ts` smart contracts.", "type": "shell", - "command": "npx scrypt-cli@latest compile --watch", + "command": "npx scrypt-cli compile --watch", "isBackground": true, "problemMatcher": [ "$tsc", From 5752f751f6399e28775e2bc15f087717cdb32d8d Mon Sep 17 00:00:00 2001 From: aaron67 Date: Tue, 26 Mar 2024 03:58:12 +0800 Subject: [PATCH 5/8] Add test --- tests/oracleDemoBsv20.test.ts | 85 +++++++++++++++++++++++++++++++++++ 1 file changed, 85 insertions(+) create mode 100644 tests/oracleDemoBsv20.test.ts diff --git a/tests/oracleDemoBsv20.test.ts b/tests/oracleDemoBsv20.test.ts new file mode 100644 index 00000000..788ce630 --- /dev/null +++ b/tests/oracleDemoBsv20.test.ts @@ -0,0 +1,85 @@ +import { expect, use } from 'chai' +import { ByteString, MethodCallOptions, bsv, toByteString } from 'scrypt-ts' +import { OracleDemoBsv20 } from '../src/contracts/oracleDemoBsv20' +import { getDefaultSigner } from './utils/helper' +import { RabinPubKey, RabinSig, WitnessOnChainVerifier } from 'scrypt-ts-lib' +import chaiAsPromised from 'chai-as-promised' +use(chaiAsPromised) + +// All data was pre-fetched from https://api.witnessonchain.com/ + +// https://api.witnessonchain.com/#/info/AppController_getInfo +const PUBKEY = { + publicKey: + 'ad7e1e8d6d2960129c9fe6b636ef4041037f599c807ecd5adf491ce45835344b18fd4e7c92fd63bb822b221344fe21c0522ab81e9f8e848206875370cae4d908ac2656192ad6910ebb685036573b442ec1cff490c1638b7f5a181ae6d6bc9a04a305720559c893611f836321c2beb69dbf3694b9305a988c77e0a451c38674e84ce95a912833d2cf4ca9d48cc76d8250d0130740145ca19e20b1513bb93ca7665c1f110493d1b5aa344702109df5feca790f988eaa02f92e019721ae0e8bfaa9fdcd3401ffb4433fbe6e575ed9f704a6dc60872f0d23b2f43bfe5e64ce0fbc71283e6dedee79e20ad878917fa4a8257f879527c58f89a8670be591fc2815f7e7a8d74a9830788404f66170058dd7a08f47c4954324088dbed2f330015ccc36d29efd392a3cd5bf9835871f6b4b203c228af16f5b461676ce8e51003afd3137978117cf41147f2bb615a7c338bebdca5f81a43fe9b51480ae52ce04cf2f2b1714599fe09ae8401e0e155b4caa89fb37b00c604517fc36961f84901a73a343bb40', +} +// https://api.witnessonchain.com/#/v1/V1Controller_getInscription +const RESP = { + timestamp: 1711167431, + outpoint: + '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557_0', + bsv20: true, + amt: 10000, + id: '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557_0', + data: '04c757fe65007972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe595255700000000011027000000000000373937326438373261653563626434666531646435393132613763656637393734353935306263653338626136663164663931326565316665353935323535375f30', + signature: { + s: '3b8f6fc2981eac44d8284b149e14acfd98a2b2c104e32c184cb01e1c6206160a2e767488db1754d84842ab664d3027147d2123875a380dd901b736b77db6a279a6f513f2b3ad2daecd44ea54ff920cb99c52da395315c018ba92b0e5e767bbdd5d64249cf535dcada6c8a01b45233f406892604945a29af3e2be8b62ed1f7d7b4415ce7f1081da64aee51eec3464b7a558ac689a834a2d9f31ecac29647221c2d9a757845e5cb621cf2c5c6e8005c4ecb18c98cb4ba30d5850c403e22e42edb064cbf43d7f20d21febc2c36480e2e7ae1a7ef14567f0e4f794600a8004d25729b96cee269409d86ed725c8380db54cc936b46c4512839287bdc3be9226f8b5378e15443761d0d4c33ba94fac1b6175ab57d82861adb88041d8a6635dcca82c1691cd600498681a8f0eaf1c8ceaf4a000edf35a5c6baed5534c43a614138cec9ddcbb2948760cd6f63c7c749bd63788e5eb530ef4461da6f6e9ca46eae374daa7f36dab316e57910e5256c09ff674bd8ad20ab06a98a014f891e7cc17da764806', + padding: '00', + }, +} + +describe('Test SmartContract `OracleDemoBsv20`', () => { + // token utxo + const txid = + '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557' + const vout = 0 + + let instance: OracleDemoBsv20 + + before(async () => { + OracleDemoBsv20.loadArtifact() + const rabinPubKey: RabinPubKey = + WitnessOnChainVerifier.parsePubKey(PUBKEY) + const inscriptionId = toByteString(`${txid}_${vout}`, true) + const amt = 10n + instance = new OracleDemoBsv20(rabinPubKey, inscriptionId, amt) + await instance.connect(getDefaultSigner()) + await instance.deploy() + }) + + it('should pass the public method unit test successfully.', async () => { + // cutomize the tx builder for public method "unlock" + instance.bindTxBuilder( + 'unlock', + async ( + current: OracleDemoBsv20, + options: MethodCallOptions + ) => { + const defaultAddress = await current.signer.getDefaultAddress() + const unsignedTx: bsv.Transaction = new bsv.Transaction() + // add contract input + .addInput(current.buildContractInput()) + // add token input + .from({ + txId: txid, + outputIndex: vout, + script: bsv.Script.fromASM('OP_TRUE').toString(), // fake script just for demo + satoshis: 1, + }) + .change(options.changeAddress || defaultAddress) + return Promise.resolve({ + tx: unsignedTx, + atInputIndex: 0, + nexts: [], + }) + } + ) + // parse the response from the oracle + const oracleMsg: ByteString = WitnessOnChainVerifier.parseMsg(RESP) + const oracleSig: RabinSig = WitnessOnChainVerifier.parseSig(RESP) + // call the public method "unlock" + const callContract = async () => + instance.methods.unlock(oracleMsg, oracleSig, 1n) + return expect(callContract()).not.rejected + }) +}) From b5d96381781c386e14360def164ea1322183795d Mon Sep 17 00:00:00 2001 From: aaron67 Date: Tue, 26 Mar 2024 04:21:51 +0800 Subject: [PATCH 6/8] Add standard contracts p2pkh and p2pk --- package-lock.json | 14 ++++---- package.json | 2 +- tests/standardContracts.test.ts | 60 +++++++++++++++++++++++++++++++++ 3 files changed, 68 insertions(+), 8 deletions(-) create mode 100644 tests/standardContracts.test.ts diff --git a/package-lock.json b/package-lock.json index 89f48f72..948f5032 100644 --- a/package-lock.json +++ b/package-lock.json @@ -11,7 +11,7 @@ "dependencies": { "paillier-bigint": "^3.4.3", "scrypt-ord": "^1.0.16", - "scrypt-ts": "^1.3.28", + "scrypt-ts": "^1.3.30", "scrypt-ts-lib": "^0.1.27" }, "devDependencies": { @@ -4734,9 +4734,9 @@ } }, "node_modules/scrypt-ts": { - "version": "1.3.28", - "resolved": "https://registry.npmjs.org/scrypt-ts/-/scrypt-ts-1.3.28.tgz", - "integrity": "sha512-joNGvZfUgXZPO1eYbSWD+pgU/jmtoOGEQACqZsuh25x9uegvEXXcQ6FcPLygHU/aItxgqITnFUhxd9GoA9t3jA==", + "version": "1.3.30", + "resolved": "https://registry.npmjs.org/scrypt-ts/-/scrypt-ts-1.3.30.tgz", + "integrity": "sha512-nZ3DJWD+9ziZ+OjT7tvG7BKpSZE6CKpMCIA9J0NWiSfyLwj3wU9wuIB+kH7F55QjgsyKsDoYh6O4pQwF99wn5Q==", "dependencies": { "deep-equal": "^2.2.0", "fast-diff": "^1.2.0", @@ -9356,9 +9356,9 @@ } }, "scrypt-ts": { - "version": "1.3.28", - "resolved": "https://registry.npmjs.org/scrypt-ts/-/scrypt-ts-1.3.28.tgz", - "integrity": "sha512-joNGvZfUgXZPO1eYbSWD+pgU/jmtoOGEQACqZsuh25x9uegvEXXcQ6FcPLygHU/aItxgqITnFUhxd9GoA9t3jA==", + "version": "1.3.30", + "resolved": "https://registry.npmjs.org/scrypt-ts/-/scrypt-ts-1.3.30.tgz", + "integrity": "sha512-nZ3DJWD+9ziZ+OjT7tvG7BKpSZE6CKpMCIA9J0NWiSfyLwj3wU9wuIB+kH7F55QjgsyKsDoYh6O4pQwF99wn5Q==", "requires": { "deep-equal": "^2.2.0", "fast-diff": "^1.2.0", diff --git a/package.json b/package.json index 8872f8dc..9d619550 100644 --- a/package.json +++ b/package.json @@ -30,7 +30,7 @@ "dependencies": { "paillier-bigint": "^3.4.3", "scrypt-ord": "^1.0.16", - "scrypt-ts": "^1.3.28", + "scrypt-ts": "^1.3.30", "scrypt-ts-lib": "^0.1.27" }, "devDependencies": { diff --git a/tests/standardContracts.test.ts b/tests/standardContracts.test.ts new file mode 100644 index 00000000..38f2fd40 --- /dev/null +++ b/tests/standardContracts.test.ts @@ -0,0 +1,60 @@ +import { expect, use } from 'chai' +import chaiAsPromised from 'chai-as-promised' +import { + findSig, + MethodCallOptions, + PubKey, + bsv, + toHex, + pubKey2Addr, + P2PKH, + P2PK, +} from 'scrypt-ts' +import { getDefaultSigner } from './utils/helper' + +use(chaiAsPromised) + +describe('Test Standard Contracts', () => { + it('P2PKH', async () => { + const privateKey = bsv.PrivateKey.fromRandom(bsv.Networks.testnet) + const publicKey = privateKey.toPublicKey() + const pubKey = PubKey(toHex(publicKey)) + // create an P2PKH instance + const instance = new P2PKH(pubKey2Addr(pubKey)) + // connect the contract instance to a signer + await instance.connect(getDefaultSigner(privateKey)) + // deploy the contract + await instance.deploy() + // call the P2PKH contract + const callContract = async () => + instance.methods.unlock( + (sigResps) => findSig(sigResps, publicKey), + pubKey, + { + pubKeyOrAddrToSign: publicKey, + } as MethodCallOptions + ) + return expect(callContract()).not.rejected + }) + + it('P2PK', async () => { + const privateKey = bsv.PrivateKey.fromRandom(bsv.Networks.testnet) + const publicKey = privateKey.toPublicKey() + const pubKey = PubKey(toHex(publicKey)) + // create an P2PK instance + const instance = new P2PK(pubKey) + // connect the contract instance to a signer + await instance.connect(getDefaultSigner(privateKey)) + // deploy the contract + await instance.deploy() + // call the P2PK contract + const callContract = async () => + instance.methods.unlock( + (sigResps) => findSig(sigResps, publicKey), + { + pubKeyOrAddrToSign: publicKey, + } as MethodCallOptions + ) + return expect(callContract()).not.rejected + }) +}) From a5f671f4d7cc58cd467ce7219264660c6eec4f10 Mon Sep 17 00:00:00 2001 From: aaron67 Date: Wed, 27 Mar 2024 05:26:03 +0800 Subject: [PATCH 7/8] Update test --- src/contracts/oracleDemoBsv20.ts | 4 +- tests/oracleDemoBsv20.test.ts | 120 +++++++++++++++++++++++-------- 2 files changed, 94 insertions(+), 30 deletions(-) diff --git a/src/contracts/oracleDemoBsv20.ts b/src/contracts/oracleDemoBsv20.ts index 3e3bd9cd..7eee4157 100644 --- a/src/contracts/oracleDemoBsv20.ts +++ b/src/contracts/oracleDemoBsv20.ts @@ -18,7 +18,7 @@ type Msg = { vout: bigint // 4 bytes LE, output index bsv20: bigint // 1 byte, token type, 0 for NFT, 1 for BSV20 amt: bigint // 8 bytes LE - id: ByteString // 66 bytes + id: ByteString } export class OracleDemoBsv20 extends SmartContract { @@ -51,7 +51,7 @@ export class OracleDemoBsv20 extends SmartContract { vout: Utils.fromLEUnsigned(slice(msg, 38n, 42n)), bsv20: Utils.fromLEUnsigned(slice(msg, 42n, 43n)), amt: Utils.fromLEUnsigned(slice(msg, 43n, 51n)), - id: slice(msg, 51n, 117n), + id: slice(msg, 51n), } } diff --git a/tests/oracleDemoBsv20.test.ts b/tests/oracleDemoBsv20.test.ts index 788ce630..121e5d93 100644 --- a/tests/oracleDemoBsv20.test.ts +++ b/tests/oracleDemoBsv20.test.ts @@ -1,13 +1,22 @@ import { expect, use } from 'chai' -import { ByteString, MethodCallOptions, bsv, toByteString } from 'scrypt-ts' +import { + Addr, + ByteString, + MethodCallOptions, + PubKey, + SmartContract, + bsv, + findSig, + toByteString, +} from 'scrypt-ts' import { OracleDemoBsv20 } from '../src/contracts/oracleDemoBsv20' import { getDefaultSigner } from './utils/helper' import { RabinPubKey, RabinSig, WitnessOnChainVerifier } from 'scrypt-ts-lib' import chaiAsPromised from 'chai-as-promised' +import { BSV20V2P2PKH } from 'scrypt-ord' use(chaiAsPromised) // All data was pre-fetched from https://api.witnessonchain.com/ - // https://api.witnessonchain.com/#/info/AppController_getInfo const PUBKEY = { publicKey: @@ -33,53 +42,108 @@ describe('Test SmartContract `OracleDemoBsv20`', () => { const txid = '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557' const vout = 0 + const script = + '0063036f726451126170706c69636174696f6e2f6273762d3230004c747b2270223a226273762d3230222c226f70223a227472616e73666572222c226964223a22643563663365373239653766363866313630646261643763623363656330303831323765353438343438346436386261326463656264336262663966613737365f30222c22616d74223a22313030227d6876a914700cc86d386b5c4707c06c96985f57ca875266e988ac' + // keys to unlock token utxo + const tokenPrivKey = bsv.PrivateKey.fromWIF( + 'cRmsBM2joHToN2fEWWh5eXuSGCinmyG7rSv1d9ZECKcngBXJnWQw' + ) + const tokenPubKey = tokenPrivKey.publicKey - let instance: OracleDemoBsv20 + let demoInstance: OracleDemoBsv20 + let tokenInstance: BSV20V2P2PKH + const signer = getDefaultSigner(tokenPrivKey) before(async () => { + // setup demo instance OracleDemoBsv20.loadArtifact() const rabinPubKey: RabinPubKey = WitnessOnChainVerifier.parsePubKey(PUBKEY) const inscriptionId = toByteString(`${txid}_${vout}`, true) const amt = 10n - instance = new OracleDemoBsv20(rabinPubKey, inscriptionId, amt) - await instance.connect(getDefaultSigner()) - await instance.deploy() + demoInstance = new OracleDemoBsv20(rabinPubKey, inscriptionId, amt) + await demoInstance.connect(signer) + await demoInstance.deploy() + // setup token instance + tokenInstance = BSV20V2P2PKH.fromUTXO({ + txId: txid, + outputIndex: vout, + script, + satoshis: 1, + }) + await tokenInstance.connect(signer) }) it('should pass the public method unit test successfully.', async () => { - // cutomize the tx builder for public method "unlock" - instance.bindTxBuilder( + // customise call tx for demoInstance + demoInstance.bindTxBuilder('unlock', (current: OracleDemoBsv20) => { + const unsignedTx = new bsv.Transaction().addInput( + current.buildContractInput() + ) + return Promise.resolve({ + tx: unsignedTx, + atInputIndex: 0, + nexts: [], + }) + }) + // parse the response from the oracle + const oracleMsg: ByteString = WitnessOnChainVerifier.parseMsg(RESP) + const oracleSig: RabinSig = WitnessOnChainVerifier.parseSig(RESP) + // call demoInstance.unlock to get a partial tx + const partialTx = await demoInstance.methods.unlock( + oracleMsg, + oracleSig, + 1n, + { + multiContractCall: true, + } as MethodCallOptions + ) + // customise call tx for tokenInstance + tokenInstance.bindTxBuilder( 'unlock', async ( - current: OracleDemoBsv20, - options: MethodCallOptions + current: BSV20V2P2PKH, + options: MethodCallOptions ) => { - const defaultAddress = await current.signer.getDefaultAddress() - const unsignedTx: bsv.Transaction = new bsv.Transaction() - // add contract input - .addInput(current.buildContractInput()) - // add token input - .from({ - txId: txid, - outputIndex: vout, - script: bsv.Script.fromASM('OP_TRUE').toString(), // fake script just for demo - satoshis: 1, - }) - .change(options.changeAddress || defaultAddress) + const tokenChange = new BSV20V2P2PKH( + toByteString(current.id, true), + current.sym, + current.max, + current.dec, + Addr(options.changeAddress!.toByteString()) + ).setAmt(current.getAmt()) + const unsignedTx = options + .partialContractTx!.tx.addInput( + current.buildContractInput() + ) + .addOutput( + new bsv.Transaction.Output({ + script: tokenChange.lockingScript, + satoshis: 1, + }) + ) + .change(await current.signer.getDefaultAddress()) return Promise.resolve({ tx: unsignedTx, - atInputIndex: 0, + atInputIndex: 1, nexts: [], }) } ) - // parse the response from the oracle - const oracleMsg: ByteString = WitnessOnChainVerifier.parseMsg(RESP) - const oracleSig: RabinSig = WitnessOnChainVerifier.parseSig(RESP) - // call the public method "unlock" + // call tokenInstance.unlock to get the final tx + const finalTx = await tokenInstance.methods.unlock( + (sigResps) => findSig(sigResps, tokenPubKey), + PubKey(tokenPubKey.toHex()), + { + multiContractCall: true, + partialContractTx: partialTx, + pubKeyOrAddrToSign: tokenPubKey, + changeAddress: tokenPubKey.toAddress(), + } as MethodCallOptions + ) + // final call const callContract = async () => - instance.methods.unlock(oracleMsg, oracleSig, 1n) + SmartContract.multiContractCall(finalTx, signer) return expect(callContract()).not.rejected }) }) From f1cbfdef14800ed921a9fbb89ba41e2874e56b94 Mon Sep 17 00:00:00 2001 From: aaron67 Date: Wed, 27 Mar 2024 11:59:23 +0800 Subject: [PATCH 8/8] Update --- src/contracts/oracleDemoBsv20.ts | 34 ++++++++++---------------------- tests/oracleDemoBsv20.test.ts | 10 +++++----- 2 files changed, 15 insertions(+), 29 deletions(-) diff --git a/src/contracts/oracleDemoBsv20.ts b/src/contracts/oracleDemoBsv20.ts index 7eee4157..03353a11 100644 --- a/src/contracts/oracleDemoBsv20.ts +++ b/src/contracts/oracleDemoBsv20.ts @@ -5,7 +5,6 @@ import { assert, method, prop, - reverseByteString, slice, } from 'scrypt-ts' import { RabinPubKey, RabinSig, WitnessOnChainVerifier } from 'scrypt-ts-lib' @@ -14,9 +13,8 @@ type Msg = { marker: bigint // 1 byte, api marker timestamp: bigint // 4 bytes LE network: bigint // 1 byte, 1 for mainnet, 0 for testnet - txid: ByteString // 32 bytes, txid - vout: bigint // 4 bytes LE, output index - bsv20: bigint // 1 byte, token type, 0 for NFT, 1 for BSV20 + outpoint: ByteString // 36 bytes, txid 32 bytes LE + vout 4 bytes LE + fungible: bigint // 1 byte, token type, 1 for BSV20, 0 for NFT amt: bigint // 8 bytes LE id: ByteString } @@ -47,9 +45,8 @@ export class OracleDemoBsv20 extends SmartContract { marker: Utils.fromLEUnsigned(slice(msg, 0n, 1n)), timestamp: Utils.fromLEUnsigned(slice(msg, 1n, 5n)), network: Utils.fromLEUnsigned(slice(msg, 5n, 6n)), - txid: slice(msg, 6n, 38n), - vout: Utils.fromLEUnsigned(slice(msg, 38n, 42n)), - bsv20: Utils.fromLEUnsigned(slice(msg, 42n, 43n)), + outpoint: slice(msg, 6n, 42n), + fungible: Utils.fromLEUnsigned(slice(msg, 42n, 43n)), amt: Utils.fromLEUnsigned(slice(msg, 43n, 51n)), id: slice(msg, 51n), } @@ -58,20 +55,10 @@ export class OracleDemoBsv20 extends SmartContract { @method() public unlock(msg: ByteString, sig: RabinSig, tokenInputIndex: bigint) { // retrieve token outpoint from prevouts - const txid = reverseByteString( - slice( - this.prevouts, - tokenInputIndex * 36n, - tokenInputIndex * 36n + 32n - ), - 32n - ) - const vout = Utils.fromLEUnsigned( - slice( - this.prevouts, - tokenInputIndex * 36n + 32n, - tokenInputIndex * 36n + 36n - ) + const outpoint = slice( + this.prevouts, + tokenInputIndex * 36n, + (tokenInputIndex + 1n) * 36n ) // verify oracle signature assert( @@ -83,9 +70,8 @@ export class OracleDemoBsv20 extends SmartContract { // validate data assert(message.marker == 4n, 'incorrect oracle message type') assert(message.network == 0n, 'incorrect network') - assert(message.txid == txid, 'incorrect token txid') - assert(message.vout == vout, 'incorrect token vout') - assert(message.bsv20 == 1n, 'incorrect token type') + assert(message.outpoint == outpoint, 'incorrect token outpoint') + assert(message.fungible == 1n, 'incorrect token type') assert(message.amt >= this.amt, 'incorrect token amount') assert(message.id == this.inscriptionId, 'incorrect inscription id') // do other validations ... diff --git a/tests/oracleDemoBsv20.test.ts b/tests/oracleDemoBsv20.test.ts index 121e5d93..17daa15e 100644 --- a/tests/oracleDemoBsv20.test.ts +++ b/tests/oracleDemoBsv20.test.ts @@ -24,16 +24,16 @@ const PUBKEY = { } // https://api.witnessonchain.com/#/v1/V1Controller_getInscription const RESP = { - timestamp: 1711167431, + timestamp: 1711511226, outpoint: '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557_0', - bsv20: true, + fungible: 1, amt: 10000, id: '7972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe5952557_0', - data: '04c757fe65007972d872ae5cbd4fe1dd5912a7cef79745950bce38ba6f1df912ee1fe595255700000000011027000000000000373937326438373261653563626434666531646435393132613763656637393734353935306263653338626136663164663931326565316665353935323535375f30', + data: '04ba96036600572595e51fee12f91d6fba38ce0b954597f7cea71259dde14fbd5cae72d8727900000000011027000000000000373937326438373261653563626434666531646435393132613763656637393734353935306263653338626136663164663931326565316665353935323535375f30', signature: { - s: '3b8f6fc2981eac44d8284b149e14acfd98a2b2c104e32c184cb01e1c6206160a2e767488db1754d84842ab664d3027147d2123875a380dd901b736b77db6a279a6f513f2b3ad2daecd44ea54ff920cb99c52da395315c018ba92b0e5e767bbdd5d64249cf535dcada6c8a01b45233f406892604945a29af3e2be8b62ed1f7d7b4415ce7f1081da64aee51eec3464b7a558ac689a834a2d9f31ecac29647221c2d9a757845e5cb621cf2c5c6e8005c4ecb18c98cb4ba30d5850c403e22e42edb064cbf43d7f20d21febc2c36480e2e7ae1a7ef14567f0e4f794600a8004d25729b96cee269409d86ed725c8380db54cc936b46c4512839287bdc3be9226f8b5378e15443761d0d4c33ba94fac1b6175ab57d82861adb88041d8a6635dcca82c1691cd600498681a8f0eaf1c8ceaf4a000edf35a5c6baed5534c43a614138cec9ddcbb2948760cd6f63c7c749bd63788e5eb530ef4461da6f6e9ca46eae374daa7f36dab316e57910e5256c09ff674bd8ad20ab06a98a014f891e7cc17da764806', - padding: '00', + s: '80385ea550a64558f7d0146c7b7485b36044b20e2e291cb089864c80d344476657c291688c6459ce7037cb2ee8941c1b6176a2b9e46da429ba91564daadd72f2d5b2b9550cd0a9a78df8eda96905848fec8bfbf600dd2af2cd336b7b6d0615fe1e3ce41b7de3498fe0e0cd6e86d79c62199eea2eb514663ba8fabc3eaf6a603487c10465650b20c48b87fb5d0a1f913073f90eb2b0e9f56eb41b5518f76b8bb2cd7906c5bb0855d6f0a6fe5e6456c8a27e5868f36374733043e4cc0c6dd28e5e4cd0bdb3b7b76dbcfee5a7cfe4fc87404aa431058ef58dcf528ad0a37fc5f8ede5da557bdd0e4fc5007e4e2b502199fd794785e67e534c213ca73b0bb02bebc822cfe97b5fcbaebb45ebd875499e923b10af2d05db8ec5220afad9dd45eef04569ddb8f6bbf0e6cf53407387c87358b9ebdce084930239de1efdc130df4b05814ee22a8848436ce94b2b4d87643e0d24dd9d155dcd750f726f893763d56c530d57480a207c50ea7f165c9e33909354f340ceab752eac60d52dcc653a5ca0e23f', + padding: '0000000000000000', }, }