Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor nft with cosmos-sdk nft module #309

Merged
merged 14 commits into from
Sep 6, 2022
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ require (
github.com/tidwall/gjson v1.14.0
google.golang.org/genproto v0.0.0-20220725144611-272f38e5d71b
google.golang.org/grpc v1.48.0
google.golang.org/protobuf v1.28.0
gopkg.in/yaml.v2 v2.4.0
)

Expand Down Expand Up @@ -64,7 +65,6 @@ require (
golang.org/x/xerrors v0.0.0-20220517211312-f3a8303e98df // indirect
google.golang.org/api v0.81.0 // indirect
google.golang.org/appengine v1.6.7 // indirect
google.golang.org/protobuf v1.28.0 // indirect
sigs.k8s.io/yaml v1.3.0 // indirect
)

Expand Down
2 changes: 1 addition & 1 deletion modules/mt/client/cli/tx.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ func NewTxCmd() *cobra.Command {
return txCmd
}

// GetCmdIssueDenom is the CLI command for an IssueDenom transaction
// GetCmdIssueDenom is the CLI command for an SaveDenom transaction
func GetCmdIssueDenom() *cobra.Command {
cmd := &cobra.Command{
Use: "issue",
Expand Down
17 changes: 9 additions & 8 deletions modules/mt/types/tx.pb.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions modules/nft/client/cli/cli_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -168,11 +168,11 @@ func (s *IntegrationTestSuite) TestNft() {
s.Require().Equal(from.String(), nftItem.Owner)

//------test GetCmdQueryOwner()-------------
respType = proto.Message(&nfttypes.QueryOwnerResponse{})
respType = proto.Message(&nfttypes.QueryNFTsOfOwnerResponse{})
bz, err = nfttestutil.QueryOwnerExec(val.ClientCtx, from.String())
s.Require().NoError(err)
s.Require().NoError(val.ClientCtx.Codec.UnmarshalJSON(bz.Bytes(), respType))
ownerResp := respType.(*nfttypes.QueryOwnerResponse)
ownerResp := respType.(*nfttypes.QueryNFTsOfOwnerResponse)
s.Require().Equal(from.String(), ownerResp.Owner.Address)
s.Require().Equal(denom, ownerResp.Owner.IDCollections[0].DenomId)
s.Require().Equal(tokenID, ownerResp.Owner.IDCollections[0].TokenIds[0])
Expand Down
2 changes: 1 addition & 1 deletion modules/nft/client/cli/query.go
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ func GetCmdQueryOwner() *cobra.Command {
return err
}
queryClient := types.NewQueryClient(clientCtx)
resp, err := queryClient.Owner(context.Background(), &types.QueryOwnerRequest{
resp, err := queryClient.NFTsOfOwner(context.Background(), &types.QueryNFTsOfOwnerRequest{
DenomId: denomID,
Owner: args[0],
Pagination: pageReq,
Expand Down
6 changes: 3 additions & 3 deletions modules/nft/client/cli/tx.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ func NewTxCmd() *cobra.Command {
return txCmd
}

// GetCmdIssueDenom is the CLI command for an IssueDenom transaction
// GetCmdIssueDenom is the CLI command for an SaveDenom transaction
func GetCmdIssueDenom() *cobra.Command {
cmd := &cobra.Command{
Use: "issue [denom-id]",
Expand Down Expand Up @@ -134,7 +134,7 @@ func GetCmdIssueDenom() *cobra.Command {
return cmd
}

// GetCmdMintNFT is the CLI command for a MintNFT transaction
// GetCmdMintNFT is the CLI command for a SaveNFT transaction
func GetCmdMintNFT() *cobra.Command {
cmd := &cobra.Command{
Use: "mint [denom-id] [nft-id]",
Expand Down Expand Up @@ -332,7 +332,7 @@ func GetCmdTransferNFT() *cobra.Command {
return cmd
}

// GetCmdBurnNFT is the CLI command for sending a BurnNFT transaction
// GetCmdBurnNFT is the CLI command for sending a RemoveNFT transaction
func GetCmdBurnNFT() *cobra.Command {
cmd := &cobra.Command{
Use: "burn [denom-id] [nft-id]",
Expand Down
27 changes: 24 additions & 3 deletions modules/nft/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,18 +14,39 @@ func InitGenesis(ctx sdk.Context, k keeper.Keeper, data types.GenesisState) {
}

for _, c := range data.Collections {
if err := k.SetDenom(ctx, c.Denom); err != nil {
creator, err := sdk.AccAddressFromBech32(c.Denom.Creator)
if err != nil {
panic(err)
}
if err := k.SetCollection(ctx, c); err != nil {
if err := k.SaveDenom(ctx,
c.Denom.Id,
c.Denom.Name,
c.Denom.Schema,
c.Denom.Symbol,
creator,
c.Denom.MintRestricted,
c.Denom.UpdateRestricted,
c.Denom.Description,
c.Denom.Uri,
c.Denom.UriHash,
c.Denom.Data,
); err != nil {
panic(err)
}

if err := k.SaveCollection(ctx, c); err != nil {
panic(err)
}
}
}

// ExportGenesis returns a GenesisState for a given context and keeper.
func ExportGenesis(ctx sdk.Context, k keeper.Keeper) *types.GenesisState {
return types.NewGenesisState(k.GetCollections(ctx))
collections, err := k.GetCollections(ctx)
if err != nil {
panic(err)
}
return types.NewGenesisState(collections)
}

// DefaultGenesisState returns a default genesis state
Expand Down
102 changes: 26 additions & 76 deletions modules/nft/keeper/collection.go
Original file line number Diff line number Diff line change
@@ -1,22 +1,15 @@
package keeper

import (
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"

"github.com/cosmos/cosmos-sdk/store/prefix"
sdk "github.com/cosmos/cosmos-sdk/types"
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
"github.com/cosmos/cosmos-sdk/types/query"

"github.com/irisnet/irismod/modules/nft/exported"
"github.com/irisnet/irismod/modules/nft/types"
)

// SetCollection saves all NFTs and returns an error if there already exists
func (k Keeper) SetCollection(ctx sdk.Context, collection types.Collection) error {
// SaveCollection saves all NFTs and returns an error if there already exists
func (k Keeper) SaveCollection(ctx sdk.Context, collection types.Collection) error {
for _, nft := range collection.NFTs {
if err := k.MintNFT(
if err := k.SaveNFT(
ctx,
collection.Denom.Id,
nft.GetID(),
Expand All @@ -34,85 +27,42 @@ func (k Keeper) SetCollection(ctx sdk.Context, collection types.Collection) erro

// GetCollection returns the collection by the specified denom ID
func (k Keeper) GetCollection(ctx sdk.Context, denomID string) (types.Collection, error) {
denom, found := k.GetDenom(ctx, denomID)
if !found {
return types.Collection{}, sdkerrors.Wrapf(types.ErrInvalidDenom, "denomID %s not existed ", denomID)
denom, err := k.GetDenomInfo(ctx, denomID)
if err != nil {
return types.Collection{}, err
}

nfts := k.GetNFTs(ctx, denomID)
return types.NewCollection(denom, nfts), nil
}

// GetPaginateCollection returns the collection by the specified denom ID
func (k Keeper) GetPaginateCollection(ctx sdk.Context, request *types.QueryCollectionRequest, denomID string) (types.Collection, *query.PageResponse, error) {
denom, found := k.GetDenom(ctx, denomID)
if !found {
return types.Collection{}, nil, sdkerrors.Wrapf(types.ErrInvalidDenom, "denomID %s not existed ", denomID)
}
var nfts []exported.NFT
store := ctx.KVStore(k.storeKey)
nftStore := prefix.NewStore(store, types.KeyNFT(denomID, ""))
pageRes, err := query.Paginate(nftStore, shapePageRequest(request.Pagination), func(key []byte, value []byte) error {
var baseNFT types.BaseNFT
k.cdc.MustUnmarshal(value, &baseNFT)
nfts = append(nfts, baseNFT)
return nil
})
nfts, err := k.GetNFTs(ctx, denomID)
if err != nil {
return types.Collection{}, nil, status.Errorf(codes.InvalidArgument, "paginate: %v", err)
return types.Collection{}, err
}
return types.NewCollection(denom, nfts), pageRes, nil
return types.NewCollection(*denom, nfts), nil
}

dreamer-zq marked this conversation as resolved.
Show resolved Hide resolved
// GetCollections returns all the collections
func (k Keeper) GetCollections(ctx sdk.Context) (cs []types.Collection) {
for _, denom := range k.GetDenoms(ctx) {
nfts := k.GetNFTs(ctx, denom.Id)
cs = append(cs, types.NewCollection(denom, nfts))
func (k Keeper) GetCollections(ctx sdk.Context) (cs []types.Collection, err error) {
for _, class := range k.nk.GetClasses(ctx) {
nfts, err := k.GetNFTs(ctx, class.Id)
if err != nil {
return nil, err
}

denom, err := k.GetDenomInfo(ctx, class.Id)
if err != nil {
return nil, err
}

cs = append(cs, types.NewCollection(*denom, nfts))
}
return cs
return cs, nil
}

// GetDenomSupply returns the number of NFTs by the specified denom ID
// GetTotalSupply returns the number of NFTs by the specified denom ID
func (k Keeper) GetTotalSupply(ctx sdk.Context, denomID string) uint64 {
store := ctx.KVStore(k.storeKey)
bz := store.Get(types.KeyCollection(denomID))
if len(bz) == 0 {
return 0
}
return types.MustUnMarshalSupply(k.cdc, bz)
return k.nk.GetTotalSupply(ctx, denomID)
}

// GetTotalSupplyOfOwner returns the amount of NFTs by the specified conditions
func (k Keeper) GetTotalSupplyOfOwner(ctx sdk.Context, id string, owner sdk.AccAddress) (supply uint64) {
store := ctx.KVStore(k.storeKey)
iterator := sdk.KVStorePrefixIterator(store, types.KeyOwner(owner, id, ""))
defer iterator.Close()
for ; iterator.Valid(); iterator.Next() {
supply++
}
return supply
}

func (k Keeper) increaseSupply(ctx sdk.Context, denomID string) {
supply := k.GetTotalSupply(ctx, denomID)
supply++

store := ctx.KVStore(k.storeKey)
bz := types.MustMarshalSupply(k.cdc, supply)
store.Set(types.KeyCollection(denomID), bz)
}

func (k Keeper) decreaseSupply(ctx sdk.Context, denomID string) {
supply := k.GetTotalSupply(ctx, denomID)
supply--

store := ctx.KVStore(k.storeKey)
if supply == 0 {
store.Delete(types.KeyCollection(denomID))
return
}

bz := types.MustMarshalSupply(k.cdc, supply)
store.Set(types.KeyCollection(denomID), bz)
return k.nk.GetBalance(ctx, id, owner)
}
26 changes: 13 additions & 13 deletions modules/nft/keeper/collection_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ func (suite *KeeperSuite) TestSetCollection() {
NFTs: []types.BaseNFT{nft2, nft},
}

err := suite.keeper.SetCollection(suite.ctx, collection2)
err := suite.keeper.SaveCollection(suite.ctx, collection2)
suite.Nil(err)

collection2, err = suite.keeper.GetCollection(suite.ctx, denomID)
Expand All @@ -37,8 +37,8 @@ func (suite *KeeperSuite) TestSetCollection() {
}

func (suite *KeeperSuite) TestGetCollection() {
// MintNFT shouldn't fail when collection does not exist
err := suite.keeper.MintNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
// SaveNFT shouldn't fail when collection does not exist
err := suite.keeper.SaveNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
suite.NoError(err)

// collection should exist
Expand All @@ -52,25 +52,25 @@ func (suite *KeeperSuite) TestGetCollection() {

func (suite *KeeperSuite) TestGetCollections() {

// MintNFT shouldn't fail when collection does not exist
err := suite.keeper.MintNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
// SaveNFT shouldn't fail when collection does not exist
err := suite.keeper.SaveNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
suite.NoError(err)

msg, fail := keeper.SupplyInvariant(suite.keeper)(suite.ctx)
suite.False(fail, msg)
}

func (suite *KeeperSuite) TestGetSupply() {
// MintNFT shouldn't fail when collection does not exist
err := suite.keeper.MintNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
// SaveNFT shouldn't fail when collection does not exist
err := suite.keeper.SaveNFT(suite.ctx, denomID, tokenID, tokenNm, tokenURI, tokenURIHash, tokenData, address)
suite.NoError(err)

// MintNFT shouldn't fail when collection does not exist
err = suite.keeper.MintNFT(suite.ctx, denomID, tokenID2, tokenNm2, tokenURI, tokenURIHash, tokenData, address2)
// SaveNFT shouldn't fail when collection does not exist
err = suite.keeper.SaveNFT(suite.ctx, denomID, tokenID2, tokenNm2, tokenURI, tokenURIHash, tokenData, address2)
suite.NoError(err)

// MintNFT shouldn't fail when collection does not exist
err = suite.keeper.MintNFT(suite.ctx, denomID2, tokenID, tokenNm2, tokenURI, tokenURIHash, tokenData, address2)
// SaveNFT shouldn't fail when collection does not exist
err = suite.keeper.SaveNFT(suite.ctx, denomID2, tokenID, tokenNm2, tokenURI, tokenURIHash, tokenData, address2)
suite.NoError(err)

supply := suite.keeper.GetTotalSupply(suite.ctx, denomID)
Expand All @@ -92,7 +92,7 @@ func (suite *KeeperSuite) TestGetSupply() {
suite.Equal(uint64(1), supply)

//burn nft
err = suite.keeper.BurnNFT(suite.ctx, denomID, tokenID, address)
err = suite.keeper.RemoveNFT(suite.ctx, denomID, tokenID, address)
suite.NoError(err)

supply = suite.keeper.GetTotalSupply(suite.ctx, denomID)
Expand All @@ -102,7 +102,7 @@ func (suite *KeeperSuite) TestGetSupply() {
suite.Equal(uint64(1), supply)

//burn nft
err = suite.keeper.BurnNFT(suite.ctx, denomID, tokenID2, address2)
err = suite.keeper.RemoveNFT(suite.ctx, denomID, tokenID2, address2)
suite.NoError(err)

supply = suite.keeper.GetTotalSupply(suite.ctx, denomID)
Expand Down
Loading