2024-02-16 06:40:42 +00:00
|
|
|
package keeper
|
|
|
|
|
|
|
|
import (
|
2024-02-16 09:32:52 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
|
|
|
|
"cosmossdk.io/collections"
|
|
|
|
errorsmod "cosmossdk.io/errors"
|
|
|
|
cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types"
|
2024-02-16 06:40:42 +00:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2024-02-16 09:32:52 +00:00
|
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
2024-02-16 06:40:42 +00:00
|
|
|
|
2024-02-16 09:32:52 +00:00
|
|
|
auctiontypes "git.vdb.to/cerc-io/laconic2d/x/auction"
|
2024-02-16 06:40:42 +00:00
|
|
|
registrytypes "git.vdb.to/cerc-io/laconic2d/x/registry"
|
2024-02-16 09:32:52 +00:00
|
|
|
"git.vdb.to/cerc-io/laconic2d/x/registry/helpers"
|
2024-02-16 06:40:42 +00:00
|
|
|
)
|
|
|
|
|
2024-02-16 09:32:52 +00:00
|
|
|
// HasNameAuthority - checks if a name/authority exists.
|
|
|
|
func (k Keeper) HasNameAuthority(ctx sdk.Context, name string) (bool, error) {
|
|
|
|
has, err := k.Authorities.Has(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return has, nil
|
|
|
|
}
|
|
|
|
|
2024-02-16 06:40:42 +00:00
|
|
|
// GetNameAuthority - gets a name authority from the store.
|
2024-02-16 09:32:52 +00:00
|
|
|
func (k Keeper) GetNameAuthority(ctx sdk.Context, name string) (registrytypes.NameAuthority, error) {
|
|
|
|
authority, err := k.Authorities.Get(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, collections.ErrNotFound) {
|
|
|
|
return registrytypes.NameAuthority{}, errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Name authority not found.")
|
|
|
|
}
|
|
|
|
return registrytypes.NameAuthority{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return authority, nil
|
2024-02-16 06:40:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// HasNameRecord - checks if a name record exists.
|
|
|
|
func (k Keeper) HasNameRecord(ctx sdk.Context, crn string) bool {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetNameRecord - gets a name record from the store.
|
|
|
|
func (k Keeper) GetNameRecord(ctx sdk.Context, crn string) *registrytypes.NameRecord {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListNameRecords - get all name records.
|
|
|
|
func (k Keeper) ListNameRecords(ctx sdk.Context) []registrytypes.NameEntry {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProcessSetName creates a CRN -> Record ID mapping.
|
|
|
|
func (k Keeper) ProcessSetName(ctx sdk.Context, msg registrytypes.MsgSetName) error {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
2024-02-16 09:32:52 +00:00
|
|
|
// SaveNameAuthority creates the NameAuthority record.
|
|
|
|
func (k Keeper) SaveNameAuthority(ctx sdk.Context, name string, authority *registrytypes.NameAuthority) error {
|
|
|
|
return k.Authorities.Set(ctx, name, *authority)
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// updateBlockChangeSetForNameAuthority(ctx, codec, store, name)
|
|
|
|
}
|
|
|
|
|
2024-02-19 04:47:52 +00:00
|
|
|
// ReserveAuthority reserves a name authority.
|
2024-02-16 09:32:52 +00:00
|
|
|
func (k Keeper) ReserveAuthority(ctx sdk.Context, msg registrytypes.MsgReserveAuthority) error {
|
|
|
|
crn := fmt.Sprintf("crn://%s", msg.GetName())
|
|
|
|
parsedCrn, err := url.Parse(crn)
|
|
|
|
if err != nil {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Invalid name")
|
|
|
|
}
|
|
|
|
|
|
|
|
name := parsedCrn.Host
|
|
|
|
if fmt.Sprintf("crn://%s", name) != crn {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Invalid name")
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// if strings.Contains(name, ".") {
|
|
|
|
// return k.ProcessReserveSubAuthority(ctx, name, msg)
|
|
|
|
// }
|
|
|
|
|
|
|
|
err = k.createAuthority(ctx, name, msg.GetSigner(), true)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k Keeper) createAuthority(ctx sdk.Context, name string, owner string, isRoot bool) error {
|
|
|
|
moduleParams, err := k.GetParams(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
has, err := k.HasNameAuthority(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if has {
|
|
|
|
authority, err := k.GetNameAuthority(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if authority.Status != registrytypes.AuthorityExpired {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Name already reserved.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ownerAddress, err := sdk.AccAddressFromBech32(owner)
|
|
|
|
if err != nil {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Invalid owner address.")
|
|
|
|
}
|
|
|
|
ownerAccount := k.accountKeeper.GetAccount(ctx, ownerAddress)
|
|
|
|
if ownerAccount == nil {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrUnknownAddress, "Owner account not found.")
|
|
|
|
}
|
|
|
|
|
|
|
|
authority := registrytypes.NameAuthority{
|
|
|
|
OwnerPublicKey: getAuthorityPubKey(ownerAccount.GetPubKey()),
|
|
|
|
OwnerAddress: owner,
|
|
|
|
Height: uint64(ctx.BlockHeight()),
|
|
|
|
Status: registrytypes.AuthorityActive,
|
|
|
|
AuctionId: "",
|
|
|
|
BondId: "",
|
|
|
|
ExpiryTime: ctx.BlockTime().Add(moduleParams.AuthorityGracePeriod),
|
|
|
|
}
|
|
|
|
|
|
|
|
if isRoot && moduleParams.AuthorityAuctionEnabled {
|
|
|
|
// If auctions are enabled, clear out owner fields. They will be set after a winner is picked.
|
|
|
|
authority.OwnerAddress = ""
|
|
|
|
authority.OwnerPublicKey = ""
|
|
|
|
// Reset bond ID if required.
|
|
|
|
if authority.BondId != "" || len(authority.BondId) != 0 {
|
|
|
|
// TODO
|
|
|
|
// k.RemoveBondToAuthorityIndexEntry(ctx, authority.BondId, name)
|
|
|
|
authority.BondId = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
params := auctiontypes.Params{
|
|
|
|
CommitsDuration: moduleParams.AuthorityAuctionCommitsDuration,
|
|
|
|
RevealsDuration: moduleParams.AuthorityAuctionRevealsDuration,
|
|
|
|
CommitFee: moduleParams.AuthorityAuctionCommitFee,
|
|
|
|
RevealFee: moduleParams.AuthorityAuctionRevealFee,
|
|
|
|
MinimumBid: moduleParams.AuthorityAuctionMinimumBid,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an auction.
|
|
|
|
msg := auctiontypes.NewMsgCreateAuction(params, ownerAddress)
|
|
|
|
|
|
|
|
auction, sdkErr := k.auctionKeeper.CreateAuction(ctx, msg)
|
|
|
|
if sdkErr != nil {
|
|
|
|
return sdkErr
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// Create auction ID -> authority name index.
|
|
|
|
// k.AddAuctionToAuthorityMapping(ctx, auction.Id, name)
|
|
|
|
|
|
|
|
authority.Status = registrytypes.AuthorityUnderAuction
|
|
|
|
authority.AuctionId = auction.Id
|
|
|
|
authority.ExpiryTime = auction.RevealsEndTime.Add(moduleParams.AuthorityGracePeriod)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save name authority in store.
|
|
|
|
if err = k.SaveNameAuthority(ctx, name, &authority); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// k.InsertAuthorityExpiryQueue(ctx, name, authority.ExpiryTime)
|
|
|
|
|
|
|
|
return nil
|
2024-02-16 06:40:42 +00:00
|
|
|
}
|
|
|
|
|
2024-02-19 04:47:52 +00:00
|
|
|
func (k Keeper) SetAuthorityBond(ctx sdk.Context, msg registrytypes.MsgSetAuthorityBond) error {
|
|
|
|
name := msg.GetName()
|
|
|
|
signer := msg.GetSigner()
|
|
|
|
|
|
|
|
if has, err := k.HasNameAuthority(ctx, name); !has {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Name authority not found.")
|
|
|
|
}
|
|
|
|
|
|
|
|
authority, err := k.GetNameAuthority(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if authority.OwnerAddress != signer {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrUnauthorized, "Access denied.")
|
|
|
|
}
|
|
|
|
|
|
|
|
if has, err := k.bondKeeper.HasBond(ctx, msg.BondId); !has {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "Bond not found.")
|
|
|
|
}
|
|
|
|
|
|
|
|
bond, err := k.bondKeeper.GetBondById(ctx, msg.BondId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if bond.Owner != signer {
|
|
|
|
return errorsmod.Wrap(sdkerrors.ErrUnauthorized, "Bond owner mismatch.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// No-op if bond hasn't changed.
|
|
|
|
if authority.BondId == msg.BondId {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// Remove old bond ID mapping, if any.
|
|
|
|
// if authority.BondId != "" {
|
|
|
|
// k.RemoveBondToAuthorityIndexEntry(ctx, authority.BondId, name)
|
|
|
|
// }
|
|
|
|
|
|
|
|
// Update bond id and save name authority in store.
|
|
|
|
authority.BondId = bond.Id
|
|
|
|
if err = k.SaveNameAuthority(ctx, name, &authority); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// Add new bond ID mapping.
|
|
|
|
// k.AddBondToAuthorityIndexEntry(ctx, authority.BondId, name)
|
|
|
|
|
|
|
|
return nil
|
2024-02-16 06:40:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ProcessDeleteName removes a CRN -> Record ID mapping.
|
|
|
|
func (k Keeper) ProcessDeleteName(ctx sdk.Context, msg registrytypes.MsgDeleteNameAuthority) error {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k Keeper) GetAuthorityExpiryQueue(ctx sdk.Context) []*registrytypes.ExpiryQueueRecord {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResolveCRN resolves a CRN to a record.
|
|
|
|
func (k Keeper) ResolveCRN(ctx sdk.Context, crn string) *registrytypes.Record {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
2024-02-16 09:32:52 +00:00
|
|
|
|
|
|
|
func getAuthorityPubKey(pubKey cryptotypes.PubKey) string {
|
|
|
|
if pubKey == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return helpers.BytesToBase64(pubKey.Bytes())
|
|
|
|
}
|