diff --git a/e2e/fiat_tf_test.go b/e2e/fiat_tf_test.go index e45aaa7a..b77d7c01 100644 --- a/e2e/fiat_tf_test.go +++ b/e2e/fiat_tf_test.go @@ -2,10 +2,13 @@ package e2e_test import ( "context" + "encoding/json" + "fmt" "testing" "cosmossdk.io/math" fiattokenfactorytypes "github.com/circlefin/noble-fiattokenfactory/x/fiattokenfactory/types" + sdk "github.com/cosmos/cosmos-sdk/types" "github.com/noble-assets/noble/e2e" "github.com/strangelove-ventures/interchaintest/v8" "github.com/stretchr/testify/require" @@ -171,3 +174,602 @@ func TestFiatTFAcceptOwner(t *testing.T) { } require.Equal(t, expectedOwnerResponse.Owner, showOwnerRes.Owner) } + +func TestFiatTFUpdateMasterMinter(t *testing.T) { + if testing.Short() { + t.Skip() + } + t.Parallel() + + ctx := context.Background() + + nw := e2e.NobleSpinUp(t, ctx, true) + noble := nw.Chain + val := noble.Validators[0] + + // ACTION: Happy Path: Update Master Minter + // EXPECTED: Success; Master Minter updated + + w := interchaintest.GetAndFundTestUsers(t, ctx, "new-masterMinter-1", math.OneInt(), noble) + newMM1 := w[0] + + _, err := val.ExecTx(ctx, nw.FiatTfRoles.Owner.KeyName(), "fiat-tokenfactory", "update-master-minter", newMM1.FormattedAddress()) + require.NoError(t, err, "failed to broadcast update-master-minter message") + + showMMRes, err := e2e.ShowMasterMinter(ctx, val) + require.NoError(t, err, "failed to query show-master-minter") + expectedGetMasterMinterResponse := fiattokenfactorytypes.QueryGetMasterMinterResponse{ + MasterMinter: fiattokenfactorytypes.MasterMinter{ + Address: newMM1.FormattedAddress(), + }, + } + require.Equal(t, expectedGetMasterMinterResponse.MasterMinter, showMMRes.MasterMinter) + + // ACTION: Update Master Minter while TF is paused + // EXPECTED: Success; Master Minter updated + // Status: + // Master Minter: newMM1 + + e2e.PauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + w = interchaintest.GetAndFundTestUsers(t, ctx, "new-masterMinter-2", math.OneInt(), noble) + newMM2 := w[0] + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.Owner.KeyName(), "fiat-tokenfactory", "update-master-minter", newMM2.FormattedAddress()) + require.NoError(t, err, "failed to broadcast update-master-minter message") + + showMMRes, err = e2e.ShowMasterMinter(ctx, val) + require.NoError(t, err, "failed to query show-master-minter") + expectedGetMasterMinterResponse = fiattokenfactorytypes.QueryGetMasterMinterResponse{ + MasterMinter: fiattokenfactorytypes.MasterMinter{ + Address: newMM2.FormattedAddress(), + }, + } + require.Equal(t, expectedGetMasterMinterResponse.MasterMinter, showMMRes.MasterMinter) + + e2e.UnpauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + // ACTION: Update Master Minter from non owner account + // EXPECTED: Request fails; Master Minter not updated + // Status: + // Master Minter: newMM2 + + w = interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble) + newMM3 := w[0] + alice := w[1] + + _, err = val.ExecTx(ctx, alice.KeyName(), "fiat-tokenfactory", "update-master-minter", newMM3.FormattedAddress()) + require.ErrorContains(t, err, "you are not the owner: unauthorized") + + showMMRes, err = e2e.ShowMasterMinter(ctx, val) + require.NoError(t, err, "failed to query show-master-minter") + require.Equal(t, expectedGetMasterMinterResponse.MasterMinter, showMMRes.MasterMinter) + + // ACTION: Update Master Minter from blacklisted owner account + // EXPECTED: Success; Master Minter updated + // Status: + // Master Minter: newMM2 + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.Owner) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.Owner.KeyName(), "fiat-tokenfactory", "update-master-minter", newMM3.FormattedAddress()) + require.NoError(t, err, "failed to broadcast update-master-minter message") + + showMMRes, err = e2e.ShowMasterMinter(ctx, val) + require.NoError(t, err, "failed to query show-master-minter") + expectedGetMasterMinterResponse = fiattokenfactorytypes.QueryGetMasterMinterResponse{ + MasterMinter: fiattokenfactorytypes.MasterMinter{ + Address: newMM3.FormattedAddress(), + }, + } + + require.Equal(t, expectedGetMasterMinterResponse.MasterMinter, showMMRes.MasterMinter) + + e2e.UnblacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.Owner) + + // ACTION: Update Master Minter to blacklisted Master Minter account + // EXPECTED: Success; Master Minter updated + // Status: + // Master Minter: newMM3 + + w = interchaintest.GetAndFundTestUsers(t, ctx, "new-mm-4", math.OneInt(), noble) + newMM4 := w[0] + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, newMM4) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.Owner.KeyName(), "fiat-tokenfactory", "update-master-minter", newMM4.FormattedAddress()) + require.NoError(t, err, "failed to broadcast update-master-minter message") + + showMMRes, err = e2e.ShowMasterMinter(ctx, val) + require.NoError(t, err, "failed to query show-master-minter") + expectedGetMasterMinterResponse = fiattokenfactorytypes.QueryGetMasterMinterResponse{ + MasterMinter: fiattokenfactorytypes.MasterMinter{ + Address: newMM4.FormattedAddress(), + }, + } + require.Equal(t, expectedGetMasterMinterResponse.MasterMinter, showMMRes.MasterMinter) +} + +func TestFiatTFConfigureMinterController(t *testing.T) { + if testing.Short() { + t.Skip() + } + t.Parallel() + + ctx := context.Background() + + nw := e2e.NobleSpinUp(t, ctx, true) + noble := nw.Chain + val := noble.Validators[0] + + // ACTION: Happy path: Configure Minter Controller + // EXPECTED: Success; Minter Controller is configured with Minter + + w := interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble) + minterController1 := w[0] + minter1 := w[1] + + _, err := val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController1.FormattedAddress(), minter1.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err := e2e.ShowMinterController(ctx, val, minterController1) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController := fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter1.FormattedAddress(), + Controller: minterController1.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + // ACTION: Configure Minter Controller while TF is paused + // EXPECTED: Success; Minter Controller is configured with Minter + // Status: + // minterController1 -> minter1 + + w = interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble) + minterController2 := w[0] + minter2 := w[1] + + e2e.PauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController2.FormattedAddress(), minter2.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err = e2e.ShowMinterController(ctx, val, minterController2) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController = fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter2.FormattedAddress(), + Controller: minterController2.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + e2e.UnpauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + // ACTION: Configure Minter Controller from non Master Minter account + // EXPECTED: Request fails; Minter Controller not configured with Minter + // Status: + // minterController1 -> minter1 + // minterController2 -> minter2 + + w = interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble, noble) + minterController3 := w[0] + minter3 := w[1] + alice := w[2] + + _, err = val.ExecTx(ctx, alice.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController3.FormattedAddress(), minter3.FormattedAddress()) + require.ErrorContains(t, err, "you are not the master minter: unauthorized") + + _, err = e2e.ShowMinterController(ctx, val, minterController3) + require.Error(t, err, "successfully queried for the minter controller when it should have failed") + + // ACTION: Configure a blacklisted Minter Controller and Minter from blacklisted Master Minter account + // EXPECTED: Success; Minter Controller is configured with Minter + // Status: + // minterController1 -> minter1 + // minterController2 -> minter2 + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MasterMinter) + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, minterController2) + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, minter2) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController3.FormattedAddress(), minter3.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err = e2e.ShowMinterController(ctx, val, minterController3) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController = fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter3.FormattedAddress(), + Controller: minterController3.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + e2e.UnblacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MasterMinter) + + // ACTION: Configure an already configured Minter Controller with a new Minter + // EXPECTED: Success; Minter Controller is configured with Minter. The old minter should be disassociated + // from Minter Controller but keep its status and allowance + // Status: + // minterController1 -> minter1 + // minterController2 -> minter2 + // minterController3 -> minter3 + + // configuring minter1 to ensure allowance stays the same after assigning minterController1 a new minter + _, err = val.ExecTx(ctx, minterController1.KeyName(), "fiat-tokenfactory", "configure-minter", minter1.FormattedAddress(), "1uusdc") + require.NoError(t, err, "error configuring minter controller") + + showMinterPreUpdateMinterController, err := e2e.ShowMinters(ctx, val, minter1) + require.NoError(t, err, "failed to query show-minter") + expectedShowMinters := fiattokenfactorytypes.QueryGetMintersResponse{ + Minters: fiattokenfactorytypes.Minters{ + Address: minter1.FormattedAddress(), + Allowance: sdk.Coin{ + Denom: "uusdc", + Amount: math.OneInt(), + }, + }, + } + require.Equal(t, expectedShowMinters.Minters, showMinterPreUpdateMinterController.Minters, "configured minter and or allowance is not as expected") + + w = interchaintest.GetAndFundTestUsers(t, ctx, "minter-4", math.OneInt(), noble) + minter4 := w[0] + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController1.FormattedAddress(), minter4.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err = e2e.ShowMinterController(ctx, val, minterController1) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController = fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter4.FormattedAddress(), + Controller: minterController1.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController, "expected minter and minter controller is not as expected") + + showMinterPostUpdateMinterController, err := e2e.ShowMinters(ctx, val, minter1) + require.NoError(t, err, "failed to query show-minter") + require.Equal(t, showMinterPreUpdateMinterController.Minters, showMinterPostUpdateMinterController.Minters, "the minter should not have changed since updating the minter controller with a new minter") + + // ACTION:- Configure an already configured Minter to another Minter Controller + // EXPECTED: Success; Minter Controller is configured with new Minter. Minter can have multiple Minter Controllers. + // Status: + // minterController1 -> minter4 + // minterController2 -> minter2 + // minterController3 -> minter3 + // + // minter1 has a minting allowance but is not controlled by any minterController + + w = interchaintest.GetAndFundTestUsers(t, ctx, "minter-controller-4", math.OneInt(), noble) + minterController4 := w[0] + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController4.FormattedAddress(), minter4.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err = e2e.ShowMinterController(ctx, val, minterController4) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController = fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter4.FormattedAddress(), + Controller: minterController4.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + showMCRes, err = e2e.ShowMinterController(ctx, val, minterController1) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController = fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter4.FormattedAddress(), + Controller: minterController1.FormattedAddress(), + }, + } + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + res, _, err := val.ExecQuery(ctx, "fiat-tokenfactory", "list-minter-controller") + require.NoError(t, err, "failed to query list-minter-controller") + var listMinterController fiattokenfactorytypes.QueryAllMinterControllerResponse + _ = json.Unmarshal(res, &listMinterController) + // ignore error because `pagination` does not unmarshal + + expectedListMinterController := fiattokenfactorytypes.QueryAllMinterControllerResponse{ + MinterController: []fiattokenfactorytypes.MinterController{ + { // this minter and controller were created/assigned at genesis + Minter: nw.FiatTfRoles.Minter.FormattedAddress(), + Controller: nw.FiatTfRoles.MinterController.FormattedAddress(), + }, + { + Minter: minter4.FormattedAddress(), + Controller: minterController1.FormattedAddress(), + }, + { + Minter: minter2.FormattedAddress(), + Controller: minterController2.FormattedAddress(), + }, + { + Minter: minter3.FormattedAddress(), + Controller: minterController3.FormattedAddress(), + }, + { + Minter: minter4.FormattedAddress(), + Controller: minterController4.FormattedAddress(), + }, + }, + } + + require.ElementsMatch(t, expectedListMinterController.MinterController, listMinterController.MinterController) +} + +func TestFiatTFRemoveMinterController(t *testing.T) { + if testing.Short() { + t.Skip() + } + t.Parallel() + + ctx := context.Background() + + nw := e2e.NobleSpinUp(t, ctx, true) + noble := nw.Chain + val := noble.Validators[0] + + // ACTION: Happy path: Remove Minter Controller + // EXPECTED: Success; Minter Controller is removed + + _, err := val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "remove-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress()) + require.NoError(t, err, "error removing minter controller") + + _, err = e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.Error(t, err, "successfully queried for the minter controller when it should have failed") + + // ACTION: Remove Minter Controller while TF is paused + // EXPECTED: Success; Minter Controller is removed + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress(), nw.FiatTfRoles.Minter.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err := e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController := fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: nw.FiatTfRoles.Minter.FormattedAddress(), + Controller: nw.FiatTfRoles.MinterController.FormattedAddress(), + }, + } + + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + e2e.PauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "remove-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress()) + require.NoError(t, err, "error removing minter controller") + + _, err = e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.Error(t, err, "successfully queried for the minter controller when it should have failed") + + e2e.UnpauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + // ACTION: Remove a Minter Controller from non Master Minter account + // EXPECTED: Request fails; Minter Controller remains configured with Minter + + w := interchaintest.GetAndFundTestUsers(t, ctx, "alice", math.OneInt(), noble) + alice := w[0] + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress(), nw.FiatTfRoles.Minter.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err = e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.NoError(t, err, "failed to query show-minter-controller") + + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + _, err = val.ExecTx(ctx, alice.KeyName(), "fiat-tokenfactory", "remove-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress()) + require.ErrorContains(t, err, "you are not the master minter: unauthorized") + + showMCRes, err = e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.NoError(t, err, "failed to query show-minter-controller") + + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + // ACTION: Remove Minter Controller while MasterMinter and Minter Controller are blacklisted + // EXPECTED: Success; Minter Controller is removed + // Status: + // gw minterController -> gw minter + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MasterMinter) + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MinterController) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "remove-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress()) + require.NoError(t, err, "error removing minter controller") + + _, err = e2e.ShowMinterController(ctx, val, nw.FiatTfRoles.MinterController) + require.Error(t, err, "successfully queried for the minter controller when it should have failed") + + e2e.UnblacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MasterMinter) + + // ACTION: Remove a a non existent Minter Controller + // EXPECTED: Request fails + // Status: + // no minterController setup + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "remove-minter-controller", nw.FiatTfRoles.MinterController.FormattedAddress()) + require.ErrorContains(t, err, fmt.Sprintf("minter controller with a given address (%s) doesn't exist: user not found", nw.FiatTfRoles.MinterController.FormattedAddress())) + +} + +func TestFiatTFConfigureMinter(t *testing.T) { + if testing.Short() { + t.Skip() + } + t.Parallel() + + ctx := context.Background() + + nw := e2e.NobleSpinUp(t, ctx, true) + noble := nw.Chain + val := noble.Validators[0] + + // ACTION: Happy path: Configure minter + // EXPECTED: Success; Minter is configured with allowance + + e2e.ConfigureMinter(t, ctx, val, nw.FiatTfRoles.MinterController, nw.FiatTfRoles.Minter, 20) + + // ACTION: Configure minter while TF is paused + // EXPECTED: Request fails; Minter is not configured + + // configure new minter controller and minter + w := interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble) + minterController1 := w[0] + minter1 := w[1] + + _, err := val.ExecTx(ctx, nw.FiatTfRoles.MasterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController1.FormattedAddress(), minter1.FormattedAddress()) + require.NoError(t, err, "error configuring minter controller") + + showMCRes, err := e2e.ShowMinterController(ctx, val, minterController1) + require.NoError(t, err, "failed to query show-minter-controller") + expectedShowMinterController := fiattokenfactorytypes.QueryGetMinterControllerResponse{ + MinterController: fiattokenfactorytypes.MinterController{ + Minter: minter1.FormattedAddress(), + Controller: minterController1.FormattedAddress(), + }, + } + + require.Equal(t, expectedShowMinterController.MinterController, showMCRes.MinterController) + + e2e.PauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + allowance := int64(10) + + _, err = val.ExecTx(ctx, minterController1.KeyName(), "fiat-tokenfactory", "configure-minter", minter1.FormattedAddress(), fmt.Sprintf("%duusdc", allowance)) + require.ErrorContains(t, err, "minting is paused") + + _, err = e2e.ShowMinters(ctx, val, minter1) + require.Error(t, err, "minter found; configuring minter should not have succeeded") + + e2e.UnpauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + // ACTION: Configure minter from a minter controller not associated with the minter + // EXPECTED: Request fails; Minter is not configured with new Minter Controller but old Minter retains its allowance + // Status: + // minterController1 -> minter1 (un-configured) + // gw minterController -> gw minter + + // reconfigure minter1 to ensure balance does not change + e2e.ConfigureMinter(t, ctx, val, minterController1, minter1, allowance) + + // reconfigure minter1 with a new allownace from a minter controller not associated with the minter + differentAllowance := allowance + 99 + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MinterController.KeyName(), "fiat-tokenfactory", "configure-minter", minter1.FormattedAddress(), fmt.Sprintf("%duusdc", differentAllowance)) + require.ErrorContains(t, err, "minter address ≠ minter controller's minter address") + + showMintersRes, err := e2e.ShowMinters(ctx, val, minter1) + require.NoError(t, err, "failed to query show-minters") + expectedShowMinters := fiattokenfactorytypes.QueryGetMintersResponse{ + Minters: fiattokenfactorytypes.Minters{ + Address: minter1.FormattedAddress(), + Allowance: sdk.Coin{ + Denom: "uusdc", + Amount: math.NewInt(allowance), + }, + }, + } + require.Equal(t, expectedShowMinters.Minters, showMintersRes.Minters, "configured minter allowance is not as expected") + + // ACTION: Configure a minter is blacklisted from a blacklisted minter controller + // EXPECTED: Success; Minter is configured with allowance + // Status: + // minterController1 -> minter1 + // gw minterController -> gw minter + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, minterController1) + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, minter1) + + e2e.ConfigureMinter(t, ctx, val, minterController1, minter1, 11) +} + +func TestFiatTFRemoveMinter(t *testing.T) { + if testing.Short() { + t.Skip() + } + t.Parallel() + + ctx := context.Background() + + nw := e2e.NobleSpinUp(t, ctx, true) + noble := nw.Chain + val := noble.Validators[0] + + // ACTION: Happy path: Remove minter + // EXPECTED: Success; Minter is removed + + _, err := val.ExecTx(ctx, nw.FiatTfRoles.MinterController.KeyName(), "fiat-tokenfactory", "remove-minter", nw.FiatTfRoles.Minter.FormattedAddress()) + require.NoError(t, err, "error broadcasting removing minter") + + _, err = e2e.ShowMinters(ctx, val, nw.FiatTfRoles.Minter) + require.Error(t, err, "minter found; not successfully removed") + + // ACTION: Remove minter while TF is paused + // EXPECTED: Success; Minter is removed + + allowance := int64(10) + + // reconfigure minter + e2e.ConfigureMinter(t, ctx, val, nw.FiatTfRoles.MinterController, nw.FiatTfRoles.Minter, allowance) + + e2e.PauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MinterController.KeyName(), "fiat-tokenfactory", "remove-minter", nw.FiatTfRoles.Minter.FormattedAddress()) + require.NoError(t, err, "error broadcasting removing minter") + + _, err = e2e.ShowMinters(ctx, val, nw.FiatTfRoles.Minter) + require.Error(t, err, "minter found; not successfully removed") + + e2e.UnpauseFiatTF(t, ctx, val, nw.FiatTfRoles.Pauser) + + // ACTION: Remove minter from a minter controller not associated with the minter + // EXPECTED: Request fails; Minter is not removed + // Status: + // gw minterController -> gw minter (Removed) + + // reconfigure minter + e2e.ConfigureMinter(t, ctx, val, nw.FiatTfRoles.MinterController, nw.FiatTfRoles.Minter, allowance) + + minter1, _ := e2e.SetupMinterAndController(t, ctx, noble, val, nw.FiatTfRoles.MasterMinter, allowance) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MinterController.KeyName(), "fiat-tokenfactory", "remove-minter", minter1.FormattedAddress()) + require.ErrorContains(t, err, "minter address ≠ minter controller's minter address") + + // ensure minter still exists + showMintersRes, err := e2e.ShowMinters(ctx, val, minter1) + require.NoError(t, err, "failed to query show-minter") + expectedShowMinters := fiattokenfactorytypes.QueryGetMintersResponse{ + Minters: fiattokenfactorytypes.Minters{ + Address: minter1.FormattedAddress(), + Allowance: sdk.Coin{ + Denom: "uusdc", + Amount: math.NewInt(allowance), + }, + }, + } + require.Equal(t, expectedShowMinters.Minters, showMintersRes.Minters) + + // ACTION: Remove blacklisted minter from a blacklisted minter controller + // EXPECTED: Success; Minter is removed + // Status: + // gw minterController -> gw minter + // minterController1 -> minter1 + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.MinterController) + + e2e.BlacklistAccount(t, ctx, val, nw.FiatTfRoles.Blacklister, nw.FiatTfRoles.Minter) + + _, err = val.ExecTx(ctx, nw.FiatTfRoles.MinterController.KeyName(), "fiat-tokenfactory", "remove-minter", nw.FiatTfRoles.Minter.FormattedAddress()) + require.NoError(t, err, "error broadcasting removing minter") + + _, err = e2e.ShowMinters(ctx, val, nw.FiatTfRoles.Minter) + require.Error(t, err, "minter found; not successfully removed") +} diff --git a/e2e/utils.go b/e2e/utils.go index f562aac7..910b6522 100644 --- a/e2e/utils.go +++ b/e2e/utils.go @@ -498,9 +498,9 @@ func BlacklistAccount(t *testing.T, ctx context.Context, val *cosmos.ChainNode, require.Equal(t, expectedBlacklistResponse.Blacklisted, showBlacklistedResponse.Blacklisted) } -// unblacklistAccount unblacklists an account and then runs the `show-blacklisted` query to ensure the +// UnblacklistAccount unblacklists an account and then runs the `show-blacklisted` query to ensure the // account was successfully unblacklisted on chain -func unblacklistAccount(t *testing.T, ctx context.Context, val *cosmos.ChainNode, blacklister ibc.Wallet, unBlacklist ibc.Wallet) { +func UnblacklistAccount(t *testing.T, ctx context.Context, val *cosmos.ChainNode, blacklister ibc.Wallet, unBlacklist ibc.Wallet) { _, err := val.ExecTx(ctx, blacklister.KeyName(), "fiat-tokenfactory", "unblacklist", unBlacklist.FormattedAddress()) require.NoError(t, err, "failed to broadcast blacklist message") @@ -550,8 +550,8 @@ func UnpauseFiatTF(t *testing.T, ctx context.Context, val *cosmos.ChainNode, pau require.Equal(t, expectedUnpaused, showPausedResponse) } -// setupMinterAndController creates a minter controller and minter. It also sets up a minter with an specified allowance of `uusdc` -func setupMinterAndController(t *testing.T, ctx context.Context, noble *cosmos.CosmosChain, val *cosmos.ChainNode, masterMinter ibc.Wallet, allowance int64) (minter ibc.Wallet, minterController ibc.Wallet) { +// SetupMinterAndController creates a minter controller and minter. It also sets up a minter with an specified allowance of `uusdc` +func SetupMinterAndController(t *testing.T, ctx context.Context, noble *cosmos.CosmosChain, val *cosmos.ChainNode, masterMinter ibc.Wallet, allowance int64) (minter ibc.Wallet, minterController ibc.Wallet) { w := interchaintest.GetAndFundTestUsers(t, ctx, "default", math.OneInt(), noble, noble) minterController = w[0] minter = w[1] @@ -559,7 +559,7 @@ func setupMinterAndController(t *testing.T, ctx context.Context, noble *cosmos.C _, err := val.ExecTx(ctx, masterMinter.KeyName(), "fiat-tokenfactory", "configure-minter-controller", minterController.FormattedAddress(), minter.FormattedAddress()) require.NoError(t, err, "error configuring minter controller") - showMC, err := showMinterController(ctx, val, minterController) + showMC, err := ShowMinterController(ctx, val, minterController) require.NoError(t, err, "failed to query show-minter-controller") expectedShowMinterController := fiattokenfactorytypes.QueryGetMinterControllerResponse{ MinterController: fiattokenfactorytypes.MinterController{ @@ -569,18 +569,18 @@ func setupMinterAndController(t *testing.T, ctx context.Context, noble *cosmos.C } require.Equal(t, expectedShowMinterController.MinterController, showMC.MinterController) - configureMinter(t, ctx, val, minterController, minter, allowance) + ConfigureMinter(t, ctx, val, minterController, minter, allowance) return minter, minterController } -// configureMinter configures a minter with a specified allowance of `uusdc`. It then runs the `show-minters` query to ensure +// ConfigureMinter configures a minter with a specified allowance of `uusdc`. It then runs the `show-minters` query to ensure // the minter was properly configured -func configureMinter(t *testing.T, ctx context.Context, val *cosmos.ChainNode, minterController, minter ibc.Wallet, allowance int64) { +func ConfigureMinter(t *testing.T, ctx context.Context, val *cosmos.ChainNode, minterController, minter ibc.Wallet, allowance int64) { _, err := val.ExecTx(ctx, minterController.KeyName(), "fiat-tokenfactory", "configure-minter", minter.FormattedAddress(), fmt.Sprintf("%duusdc", allowance)) require.NoError(t, err, "error configuring minter") - showMinter, err := showMinters(ctx, val, minter) + showMinter, err := ShowMinters(ctx, val, minter) require.NoError(t, err, "failed to query show-minter") expectedShowMinters := fiattokenfactorytypes.QueryGetMintersResponse{ Minters: fiattokenfactorytypes.Minters{ @@ -595,9 +595,9 @@ func configureMinter(t *testing.T, ctx context.Context, val *cosmos.ChainNode, m require.Equal(t, expectedShowMinters.Minters, showMinter.Minters) } -// showMinterController queries for a specific minter controller by running: `query fiat-tokenfactory show-minter-controller
`. +// ShowMinterController queries for a specific minter controller by running: `query fiat-tokenfactory show-minter-controller
`. // An error is returned if the minter controller does not exist -func showMinterController(ctx context.Context, val *cosmos.ChainNode, minterController ibc.Wallet) (fiattokenfactorytypes.QueryGetMinterControllerResponse, error) { +func ShowMinterController(ctx context.Context, val *cosmos.ChainNode, minterController ibc.Wallet) (fiattokenfactorytypes.QueryGetMinterControllerResponse, error) { res, _, err := val.ExecQuery(ctx, "fiat-tokenfactory", "show-minter-controller", minterController.FormattedAddress()) if err != nil { return fiattokenfactorytypes.QueryGetMinterControllerResponse{}, err @@ -612,9 +612,9 @@ func showMinterController(ctx context.Context, val *cosmos.ChainNode, minterCont return showMinterController, nil } -// showMinters queries for a specific minter by running: `query fiat-tokenfactory show-minters
`. +// ShowMinters queries for a specific minter by running: `query fiat-tokenfactory show-minters
`. // An error is returned if the minter does not exist -func showMinters(ctx context.Context, val *cosmos.ChainNode, minter ibc.Wallet) (fiattokenfactorytypes.QueryGetMintersResponse, error) { +func ShowMinters(ctx context.Context, val *cosmos.ChainNode, minter ibc.Wallet) (fiattokenfactorytypes.QueryGetMintersResponse, error) { res, _, err := val.ExecQuery(ctx, "fiat-tokenfactory", "show-minters", minter.FormattedAddress()) if err != nil { return fiattokenfactorytypes.QueryGetMintersResponse{}, err @@ -645,8 +645,8 @@ func ShowOwner(ctx context.Context, val *cosmos.ChainNode) (fiattokenfactorytype return showOwnerResponse, nil } -// showMasterMinter queries for the token factory Master Minter by running: `query fiat-tokenfactory show-master-minter`. -func showMasterMinter(ctx context.Context, val *cosmos.ChainNode) (fiattokenfactorytypes.QueryGetMasterMinterResponse, error) { +// ShowMasterMinter queries for the token factory Master Minter by running: `query fiat-tokenfactory show-master-minter`. +func ShowMasterMinter(ctx context.Context, val *cosmos.ChainNode) (fiattokenfactorytypes.QueryGetMasterMinterResponse, error) { res, _, err := val.ExecQuery(ctx, "fiat-tokenfactory", "show-master-minter") if err != nil { return fiattokenfactorytypes.QueryGetMasterMinterResponse{}, err