diff --git a/clients/js/src/plugins/externalPluginAdapters.ts b/clients/js/src/plugins/externalPluginAdapters.ts index b276055d..11d957b0 100644 --- a/clients/js/src/plugins/externalPluginAdapters.ts +++ b/clients/js/src/plugins/externalPluginAdapters.ts @@ -1,4 +1,9 @@ -import { AccountMeta, Context, PublicKey } from '@metaplex-foundation/umi'; +import { + AccountMeta, + Context, + isSome, + PublicKey, +} from '@metaplex-foundation/umi'; import { lifecycleHookFromBase, LifecycleHookInitInfoArgs, @@ -59,7 +64,9 @@ import { export type ExternalPluginAdapterTypeString = BaseExternalPluginAdapterKey['__kind']; -export type BaseExternalPluginAdapter = BasePlugin & LifecycleChecksContainer; +export type BaseExternalPluginAdapter = BasePlugin & + ExternalPluginAdapterData & + LifecycleChecksContainer; export type ExternalPluginAdapters = | LifecycleHookPlugin @@ -125,8 +132,8 @@ export const externalPluginAdapterManifests = { }; export type ExternalPluginAdapterData = { - dataLen: bigint; - dataOffset: bigint; + dataLen?: bigint; + dataOffset?: bigint; }; export function externalRegistryRecordsToExternalPluginAdapterList( @@ -156,6 +163,10 @@ export function externalRegistryRecordsToExternalPluginAdapterList( } result.lifecycleHooks.push({ type: 'LifecycleHook', + dataOffset: isSome(record.dataOffset) + ? record.dataOffset.value + : undefined, + dataLen: isSome(record.dataLen) ? record.dataLen.value : undefined, ...mappedPlugin, ...lifecycleHookFromBase( deserializedPlugin.fields[0], @@ -169,6 +180,10 @@ export function externalRegistryRecordsToExternalPluginAdapterList( } result.appDatas.push({ type: 'AppData', + dataOffset: isSome(record.dataOffset) + ? record.dataOffset.value + : undefined, + dataLen: isSome(record.dataLen) ? record.dataLen.value : undefined, ...mappedPlugin, ...appDataFromBase(deserializedPlugin.fields[0], record, accountData), }); @@ -214,6 +229,10 @@ export function externalRegistryRecordsToExternalPluginAdapterList( } result.dataSections.push({ type: 'DataSection', + dataOffset: isSome(record.dataOffset) + ? record.dataOffset.value + : undefined, + dataLen: isSome(record.dataLen) ? record.dataLen.value : undefined, ...mappedPlugin, ...dataSectionFromBase( deserializedPlugin.fields[0], diff --git a/clients/js/test/externalPlugins/appData.test.ts b/clients/js/test/externalPlugins/appData.test.ts index 2d59c4e5..f4713b84 100644 --- a/clients/js/test/externalPlugins/appData.test.ts +++ b/clients/js/test/externalPlugins/appData.test.ts @@ -18,6 +18,8 @@ import { create, createCollection, updateCollectionPlugin, + removePlugin, + removeExternalPluginAdapterV1, } from '../../src'; const DATA_AUTHORITIES: PluginAuthorityType[] = [ @@ -832,3 +834,489 @@ test('it cannot update app data on collection using update authority when differ ], }); }); + +test('Data offsets are correctly bumped when removing other plugins', async (t) => { + const umi = await createUmi(); + const asset = await createAsset(umi, { + plugins: [ + { type: 'FreezeDelegate', frozen: false }, + { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: { + frozen: false, + authority: { type: 'Owner' }, + offset: 119n, + }, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 121n, + dataOffset: 124n, + dataLen: 4n, + }, + ], + }); + + await removePlugin(umi, { + asset: asset.publicKey, + plugin: { type: 'FreezeDelegate' }, + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when moving other external plugins', async (t) => { + const umi = await createUmi(); + const asset = await createAsset(umi, { + plugins: [ + { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([]), + offset: 119n, + dataOffset: 122n, + dataLen: 0n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 122n, + dataOffset: 125n, + dataLen: 4n, + }, + ], + }); + + await removeExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + key: { + __kind: 'AppData', + fields: [{ __kind: 'Owner' }], + }, + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when removing other external plugins with data', async (t) => { + const umi = await createUmi(); + const asset = await createAsset(umi, { + plugins: [ + { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 126n, + dataOffset: 129n, + dataLen: 4n, + }, + ], + }); + + await removeExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + key: { + __kind: 'AppData', + fields: [{ __kind: 'Owner' }], + }, + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when rewriting other external plugins to be smaller', async (t) => { + const umi = await createUmi(); + const asset = await createAsset(umi, { + plugins: [ + { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 126n, + dataOffset: 129n, + dataLen: 4n, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2]), + offset: 119n, + dataOffset: 122n, + dataLen: 2n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 124n, + dataOffset: 127n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when rewriting other external plugins to be larger', async (t) => { + const umi = await createUmi(); + const asset = await createAsset(umi, { + plugins: [ + { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 126n, + dataOffset: 129n, + dataLen: 4n, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4, 1, 2]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Address', address: umi.identity.publicKey }, + freezeDelegate: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4, 1, 2]), + offset: 119n, + dataOffset: 122n, + dataLen: 6n, + }, + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 128n, + dataOffset: 131n, + dataLen: 4n, + }, + ], + }); +}); diff --git a/clients/js/test/externalPlugins/linkedAppData.test.ts b/clients/js/test/externalPlugins/linkedAppData.test.ts index 6caa942a..d324a379 100644 --- a/clients/js/test/externalPlugins/linkedAppData.test.ts +++ b/clients/js/test/externalPlugins/linkedAppData.test.ts @@ -20,6 +20,8 @@ import { createCollection, updateCollectionPlugin, addPlugin, + addExternalPluginAdapterV1, + removeExternalPluginAdapterV1, } from '../../src'; const DATA_AUTHORITIES: PluginAuthorityType[] = [ @@ -924,3 +926,369 @@ test('it cannot update linked app data on collection using update authority when ], }); }); + +test('Data offsets are correctly bumped when removing Data Section with data', async (t) => { + const umi = await createUmi(); + const { asset, collection } = await createAssetWithCollection( + umi, + {}, + { + plugins: [ + { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + } + ); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await addExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + initInfo: { + __kind: 'AppData', + fields: [ + { + dataAuthority: { __kind: 'UpdateAuthority' }, + initPluginAuthority: null, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }, + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + dataSections: [ + { + type: 'DataSection', + parentKey: { type: 'LinkedAppData', dataAuthority: { type: 'Owner' } }, + authority: { type: 'None' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 123n, + dataLen: 4n, + }, + ], + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 127n, + dataOffset: 130n, + dataLen: 4n, + }, + ], + }); + + await removeExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + __kind: 'DataSection', + fields: [{ __kind: 'LinkedAppData', fields: [{ __kind: 'Owner' }] }], + }, + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + freezeDelegate: undefined, + dataSections: undefined, + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 119n, + dataOffset: 122n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when rewriting Data Section to be smaller', async (t) => { + const umi = await createUmi(); + const { asset, collection } = await createAssetWithCollection( + umi, + {}, + { + plugins: [ + { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + } + ); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await addExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + initInfo: { + __kind: 'AppData', + fields: [ + { + dataAuthority: { __kind: 'UpdateAuthority' }, + initPluginAuthority: null, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }, + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + dataSections: [ + { + type: 'DataSection', + parentKey: { type: 'LinkedAppData', dataAuthority: { type: 'Owner' } }, + authority: { type: 'None' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 123n, + dataLen: 4n, + }, + ], + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 127n, + dataOffset: 130n, + dataLen: 4n, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + dataSections: [ + { + type: 'DataSection', + parentKey: { type: 'LinkedAppData', dataAuthority: { type: 'Owner' } }, + authority: { type: 'None' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2]), + offset: 119n, + dataOffset: 123n, + dataLen: 2n, + }, + ], + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 125n, + dataOffset: 128n, + dataLen: 4n, + }, + ], + }); +}); + +test('Data offsets are correctly bumped when rewriting other Data Section to be larger', async (t) => { + const umi = await createUmi(); + const { asset, collection } = await createAssetWithCollection( + umi, + {}, + { + plugins: [ + { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + } + ); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4]), + }).sendAndConfirm(umi); + + await addExternalPluginAdapterV1(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + initInfo: { + __kind: 'AppData', + fields: [ + { + dataAuthority: { __kind: 'UpdateAuthority' }, + initPluginAuthority: null, + schema: ExternalPluginAdapterSchema.Binary, + }, + ], + }, + }).sendAndConfirm(umi); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'AppData', + dataAuthority: { type: 'UpdateAuthority' }, + }, + data: new Uint8Array([5, 6, 7, 8]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + dataSections: [ + { + type: 'DataSection', + parentKey: { type: 'LinkedAppData', dataAuthority: { type: 'Owner' } }, + authority: { type: 'None' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4]), + offset: 119n, + dataOffset: 123n, + dataLen: 4n, + }, + ], + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 127n, + dataOffset: 130n, + dataLen: 4n, + }, + ], + }); + + await writeData(umi, { + asset: asset.publicKey, + collection: collection.publicKey, + key: { + type: 'LinkedAppData', + dataAuthority: { type: 'Owner' }, + }, + data: new Uint8Array([1, 2, 3, 4, 1, 2]), + }).sendAndConfirm(umi); + + await assertAsset(t, umi, { + ...DEFAULT_ASSET, + asset: asset.publicKey, + owner: umi.identity.publicKey, + updateAuthority: { type: 'Collection', address: collection.publicKey }, + dataSections: [ + { + type: 'DataSection', + parentKey: { type: 'LinkedAppData', dataAuthority: { type: 'Owner' } }, + authority: { type: 'None' }, + dataAuthority: { type: 'Owner' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([1, 2, 3, 4, 1, 2]), + offset: 119n, + dataOffset: 123n, + dataLen: 6n, + }, + ], + appDatas: [ + { + type: 'AppData', + authority: { type: 'UpdateAuthority' }, + dataAuthority: { type: 'UpdateAuthority' }, + schema: ExternalPluginAdapterSchema.Binary, + data: new Uint8Array([5, 6, 7, 8]), + offset: 129n, + dataOffset: 132n, + dataLen: 4n, + }, + ], + }); +}); diff --git a/programs/mpl-core/src/plugins/plugin_registry.rs b/programs/mpl-core/src/plugins/plugin_registry.rs index f45b4813..046f1813 100644 --- a/programs/mpl-core/src/plugins/plugin_registry.rs +++ b/programs/mpl-core/src/plugins/plugin_registry.rs @@ -87,20 +87,22 @@ impl PluginRegistryV1 { for record in &mut self.external_registry { if record.offset > offset { + solana_program::msg!("Bumping Record: {:?}", record); record.offset = (record.offset as isize) .checked_add(size_diff) .ok_or(MplCoreError::NumericalOverflow)? as usize; - } - if let Some(data_offset) = record.data_offset { - if data_offset > offset { - record.data_offset = Some( - (data_offset as isize) - .checked_add(size_diff) - .ok_or(MplCoreError::NumericalOverflow)? - as usize, - ); + if let Some(data_offset) = record.data_offset { + if data_offset > offset { + solana_program::msg!("Bumping Data: {:?}", record); + record.data_offset = Some( + (data_offset as isize) + .checked_add(size_diff) + .ok_or(MplCoreError::NumericalOverflow)? + as usize, + ); + } } } } diff --git a/programs/mpl-core/src/plugins/utils.rs b/programs/mpl-core/src/plugins/utils.rs index 5dd60a9a..66d311d8 100644 --- a/programs/mpl-core/src/plugins/utils.rs +++ b/programs/mpl-core/src/plugins/utils.rs @@ -490,7 +490,7 @@ pub fn update_external_plugin_adapter_data<'a, T: DataBlob + SolanaAccount>( .ok_or(MplCoreError::NumericalOverflow)?; // Update any offsets that will change. - plugin_registry.bump_offsets(data_offset, size_diff)?; + plugin_registry.bump_offsets(record.offset, size_diff)?; let new_registry_offset = (plugin_header.plugin_registry_offset as isize) .checked_add(size_diff) @@ -633,17 +633,7 @@ pub fn delete_plugin<'a, T: DataBlob>( header.save(account, asset.get_size())?; // Move offsets for existing registry records. - for record in &mut plugin_registry.registry { - if plugin_offset < record.offset { - record.offset -= serialized_plugin.len() - } - } - - for record in &mut plugin_registry.external_registry { - if plugin_offset < record.offset { - record.offset -= serialized_plugin.len() - } - } + plugin_registry.bump_offsets(plugin_offset, -(serialized_plugin.len() as isize))?; plugin_registry.save(account, new_registry_offset)?; @@ -681,10 +671,14 @@ pub fn delete_external_plugin_adapter<'a, T: DataBlob>( let plugin_offset = registry_record.offset; let plugin = ExternalPluginAdapter::load(account, plugin_offset)?; let serialized_plugin = plugin.try_to_vec()?; + let serialized_plugin_len = serialized_plugin + .len() + .checked_add(registry_record.data_len.unwrap_or(0)) + .ok_or(MplCoreError::NumericalOverflow)?; // Get the offset of the plugin after the one being removed. let next_plugin_offset = plugin_offset - .checked_add(serialized_plugin.len()) + .checked_add(serialized_plugin_len) .ok_or(MplCoreError::NumericalOverflow)?; // Calculate the new size of the account. @@ -692,12 +686,12 @@ pub fn delete_external_plugin_adapter<'a, T: DataBlob>( .data_len() .checked_sub(serialized_registry_record.len()) .ok_or(MplCoreError::NumericalOverflow)? - .checked_sub(serialized_plugin.len()) + .checked_sub(serialized_plugin_len) .ok_or(MplCoreError::NumericalOverflow)?; let new_registry_offset = header .plugin_registry_offset - .checked_sub(serialized_plugin.len()) + .checked_sub(serialized_plugin_len) .ok_or(MplCoreError::NumericalOverflow)?; let data_to_move = header @@ -717,17 +711,7 @@ pub fn delete_external_plugin_adapter<'a, T: DataBlob>( header.save(account, asset.get_size())?; // Move offsets for existing registry records. - for record in &mut plugin_registry.external_registry { - if plugin_offset < record.offset { - record.offset -= serialized_plugin.len() - } - } - - for record in &mut plugin_registry.registry { - if plugin_offset < record.offset { - record.offset -= serialized_plugin.len() - } - } + plugin_registry.bump_offsets(plugin_offset, -(serialized_plugin_len as isize))?; plugin_registry.save(account, new_registry_offset)?; diff --git a/programs/mpl-core/src/processor/update.rs b/programs/mpl-core/src/processor/update.rs index e51f3d38..23c6d9db 100644 --- a/programs/mpl-core/src/processor/update.rs +++ b/programs/mpl-core/src/processor/update.rs @@ -368,21 +368,7 @@ fn process_update<'a, T: DataBlob + SolanaAccount>( plugin_header.save(account, new_core_size as usize)?; // Move offsets for existing registry records. - for record in &mut plugin_registry.external_registry { - let new_offset = (record.offset as isize) - .checked_add(size_diff) - .ok_or(MplCoreError::NumericalOverflow)?; - - record.offset = new_offset as usize; - } - - for record in &mut plugin_registry.registry { - let new_offset = (record.offset as isize) - .checked_add(size_diff) - .ok_or(MplCoreError::NumericalOverflow)?; - - record.offset = new_offset as usize; - } + plugin_registry.bump_offsets(core_size as usize, size_diff)?; plugin_registry.save(account, new_registry_offset as usize)?; } else {