From 39c44fa282a3008a08f97985d7e0a493bb0d63a6 Mon Sep 17 00:00:00 2001 From: Michael Danenberg <56533526+danenbm@users.noreply.github.com> Date: Fri, 3 May 2024 01:01:34 -0700 Subject: [PATCH] More tests for invalid checks and multiple oracle test --- .../js/test/externalPlugins/oracle.test.ts | 168 +++++++++++++++++- 1 file changed, 165 insertions(+), 3 deletions(-) diff --git a/clients/js/test/externalPlugins/oracle.test.ts b/clients/js/test/externalPlugins/oracle.test.ts index a27bce00..77875da7 100644 --- a/clients/js/test/externalPlugins/oracle.test.ts +++ b/clients/js/test/externalPlugins/oracle.test.ts @@ -262,7 +262,7 @@ test('it can use fixed address oracle to deny transfer', async (t) => { }); }); -test('it cannot create asset with oracle with no lifecycle checks', async (t) => { +test('it cannot create asset with oracle that has no lifecycle checks', async (t) => { const umi = await createUmi(); const account = generateSigner(umi); const owner = generateSigner(umi); @@ -347,7 +347,32 @@ test('it cannot create asset with oracle that can approve', async (t) => { await t.throwsAsync(result, { name: 'OracleCanRejectOnly' }); }); -test('it cannot add oracle that can approve to asset', async (t) => { +test('it cannot create asset with oracle that can approve in addition to reject', async (t) => { + const umi = await createUmi(); + const account = generateSigner(umi); + const owner = generateSigner(umi); + + // Oracle with `CheckResult.CAN_APPROVE` + const result = createAsset(umi, { + owner, + plugins: [ + { + type: 'Oracle', + resultsOffset: { + type: 'Anchor', + }, + lifecycleChecks: { + transfer: [CheckResult.CAN_APPROVE, CheckResult.CAN_REJECT], + }, + baseAddress: account.publicKey, + }, + ], + }); + + await t.throwsAsync(result, { name: 'OracleCanRejectOnly' }); +}); + +test('it cannot add oracle to asset that can approve', async (t) => { const umi = await createUmi(); const account = generateSigner(umi); const owner = generateSigner(umi); @@ -386,6 +411,45 @@ test('it cannot add oracle that can approve to asset', async (t) => { }); }); +test('it cannot add oracle to asset that can approve in addition to reject', async (t) => { + const umi = await createUmi(); + const account = generateSigner(umi); + const owner = generateSigner(umi); + + const asset = await createAsset(umi, { + owner, + }); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: owner.publicKey, + }); + + // Oracle with `CheckResult.CAN_APPROVE` + const result = addPlugin(umi, { + asset: asset.publicKey, + plugin: { + type: 'Oracle', + resultsOffset: { + type: 'Anchor', + }, + lifecycleChecks: { + transfer: [CheckResult.CAN_APPROVE, CheckResult.CAN_REJECT], + }, + baseAddress: account.publicKey, + }, + }).sendAndConfirm(umi); + + await t.throwsAsync(result, { name: 'OracleCanRejectOnly' }); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: owner.publicKey, + }); +}); + test('it cannot create asset with oracle that can listen', async (t) => { const umi = await createUmi(); const account = generateSigner(umi); @@ -411,7 +475,7 @@ test('it cannot create asset with oracle that can listen', async (t) => { await t.throwsAsync(result, { name: 'OracleCanRejectOnly' }); }); -test('it cannot add oracle that can listen to asset', async (t) => { +test('it cannot add oracle to asset that can listen', async (t) => { const umi = await createUmi(); const account = generateSigner(umi); const owner = generateSigner(umi); @@ -1402,3 +1466,101 @@ test('it can use preconfigured asset pda custom offset oracle to deny update', a name: 'new name 2', }); }); + +test('it can use one fixed address oracle to deny transfer when a second oracle allows it', async (t) => { + const umi = await createUmi(); + const account1 = generateSigner(umi); + const account2 = generateSigner(umi); + + // write to example program oracle account + await fixedAccountInit(umi, { + account: account1, + signer: umi.identity, + payer: umi.identity, + args: { + oracleData: { + __kind: 'V1', + create: ExternalValidationResult.Pass, + update: ExternalValidationResult.Pass, + transfer: ExternalValidationResult.Rejected, + burn: ExternalValidationResult.Pass, + }, + }, + }).sendAndConfirm(umi); + + // write to example program oracle account + await fixedAccountInit(umi, { + account: account2, + signer: umi.identity, + payer: umi.identity, + args: { + oracleData: { + __kind: 'V1', + create: ExternalValidationResult.Pass, + update: ExternalValidationResult.Pass, + transfer: ExternalValidationResult.Pass, + burn: ExternalValidationResult.Pass, + }, + }, + }).sendAndConfirm(umi); + + // create asset referencing both oracle accounts + const asset = await createAsset(umi, { + plugins: [ + { + type: 'Oracle', + resultsOffset: { + type: 'Anchor', + }, + lifecycleChecks: { + transfer: [CheckResult.CAN_REJECT], + }, + baseAddress: account1.publicKey, + }, + { + type: 'Oracle', + resultsOffset: { + type: 'Anchor', + }, + lifecycleChecks: { + transfer: [CheckResult.CAN_REJECT], + }, + baseAddress: account2.publicKey, + }, + ], + }); + + const newOwner = generateSigner(umi); + + const result = transfer(umi, { + asset, + newOwner: newOwner.publicKey, + }).sendAndConfirm(umi); + + await t.throwsAsync(result, { name: 'InvalidAuthority' }); + + await fixedAccountSet(umi, { + account: account1.publicKey, + signer: umi.identity, + args: { + oracleData: { + __kind: 'V1', + create: ExternalValidationResult.Pass, + update: ExternalValidationResult.Pass, + transfer: ExternalValidationResult.Pass, + burn: ExternalValidationResult.Pass, + }, + }, + }).sendAndConfirm(umi); + + await transfer(umi, { + asset, + newOwner: newOwner.publicKey, + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: newOwner.publicKey, + }); +});