344 lines
9.0 KiB
Go
344 lines
9.0 KiB
Go
package gql
|
|
|
|
import (
|
|
"context"
|
|
"fmt" // #nosec G702
|
|
"strconv"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/ipld/go-ipld-prime"
|
|
"github.com/ipld/go-ipld-prime/codec/dagjson"
|
|
|
|
auctiontypes "git.vdb.to/cerc-io/laconicd/x/auction"
|
|
bondtypes "git.vdb.to/cerc-io/laconicd/x/bond"
|
|
registrytypes "git.vdb.to/cerc-io/laconicd/x/registry"
|
|
)
|
|
|
|
// OwnerAttributeName denotes the owner attribute name for a bond.
|
|
const OwnerAttributeName = "owner"
|
|
|
|
// BondIDAttributeName denotes the record bond ID.
|
|
const BondIDAttributeName = "bondId"
|
|
|
|
// ExpiryTimeAttributeName denotes the record expiry time.
|
|
const ExpiryTimeAttributeName = "expiryTime"
|
|
|
|
func getGQLCoin(coin sdk.Coin) *Coin {
|
|
gqlCoin := Coin{
|
|
Type: coin.Denom,
|
|
Quantity: coin.Amount.BigInt().String(),
|
|
}
|
|
|
|
return &gqlCoin
|
|
}
|
|
|
|
func getGQLCoins(coins sdk.Coins) []*Coin {
|
|
gqlCoins := make([]*Coin, len(coins))
|
|
for index, coin := range coins {
|
|
gqlCoins[index] = getGQLCoin(coin)
|
|
}
|
|
|
|
return gqlCoins
|
|
}
|
|
|
|
func GetGQLNameAuthorityRecord(record *registrytypes.NameAuthority) (*AuthorityRecord, error) {
|
|
if record == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
return &AuthorityRecord{
|
|
OwnerAddress: record.OwnerAddress,
|
|
OwnerPublicKey: record.OwnerPublicKey,
|
|
Height: strconv.FormatUint(record.Height, 10),
|
|
Status: record.Status,
|
|
BondID: record.GetBondId(),
|
|
ExpiryTime: record.GetExpiryTime().String(),
|
|
}, nil
|
|
}
|
|
|
|
func getGQLRecord(ctx context.Context, resolver QueryResolver, record registrytypes.Record) (*Record, error) {
|
|
// Nil record.
|
|
if record.Deleted {
|
|
return nil, nil
|
|
}
|
|
|
|
node, err := ipld.Decode(record.Attributes, dagjson.Decode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if node.Kind() != ipld.Kind_Map {
|
|
return nil, fmt.Errorf("invalid record attributes")
|
|
}
|
|
|
|
var links []string
|
|
attributes, err := resolveIPLDNode(node, &links)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
references, err := resolver.GetRecordsByIds(ctx, links)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Record{
|
|
ID: record.Id,
|
|
BondID: record.GetBondId(),
|
|
CreateTime: record.GetCreateTime(),
|
|
ExpiryTime: record.GetExpiryTime(),
|
|
Owners: record.GetOwners(),
|
|
Names: record.GetNames(),
|
|
Attributes: attributes.(MapValue).Value,
|
|
References: references,
|
|
}, nil
|
|
}
|
|
|
|
func resolveIPLDNode(node ipld.Node, links *[]string) (Value, error) {
|
|
switch node.Kind() {
|
|
case ipld.Kind_Map:
|
|
var entries []*Attribute
|
|
for itr := node.MapIterator(); !itr.Done(); {
|
|
k, v, err := itr.Next()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if k.Kind() != ipld.Kind_String {
|
|
return nil, fmt.Errorf("invalid record attribute key type: %s", k.Kind())
|
|
}
|
|
s, err := k.AsString()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
val, err := resolveIPLDNode(v, links)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
entries = append(entries, &Attribute{
|
|
Key: s,
|
|
Value: val,
|
|
})
|
|
}
|
|
return MapValue{entries}, nil
|
|
case ipld.Kind_List:
|
|
var values []Value
|
|
for itr := node.ListIterator(); !itr.Done(); {
|
|
_, v, err := itr.Next()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
val, err := resolveIPLDNode(v, links)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
values = append(values, val)
|
|
}
|
|
return ArrayValue{values}, nil
|
|
case ipld.Kind_Null:
|
|
return nil, nil
|
|
case ipld.Kind_Bool:
|
|
val, err := node.AsBool()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return BooleanValue{val}, nil
|
|
case ipld.Kind_Int:
|
|
val, err := node.AsInt()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// TODO: handle bigger ints
|
|
return IntValue{int(val)}, nil
|
|
case ipld.Kind_Float:
|
|
val, err := node.AsFloat()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return FloatValue{val}, nil
|
|
case ipld.Kind_String:
|
|
val, err := node.AsString()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return StringValue{val}, nil
|
|
case ipld.Kind_Bytes:
|
|
val, err := node.AsBytes()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return BytesValue{string(val)}, nil
|
|
case ipld.Kind_Link:
|
|
val, err := node.AsLink()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
*links = append(*links, val.String())
|
|
return LinkValue{Link(val.String())}, nil
|
|
default:
|
|
return nil, fmt.Errorf("invalid node kind")
|
|
}
|
|
}
|
|
|
|
func getGQLNameRecord(record *registrytypes.NameRecord) (*NameRecord, error) {
|
|
if record == nil {
|
|
return nil, fmt.Errorf("got nil record")
|
|
}
|
|
|
|
records := make([]*NameRecordEntry, len(record.History))
|
|
for index, entry := range record.History {
|
|
records[index] = getNameRecordEntry(entry)
|
|
}
|
|
|
|
return &NameRecord{
|
|
Latest: getNameRecordEntry(record.Latest),
|
|
History: records,
|
|
}, nil
|
|
}
|
|
|
|
func getNameRecordEntry(record *registrytypes.NameRecordEntry) *NameRecordEntry {
|
|
return &NameRecordEntry{
|
|
ID: record.Id,
|
|
Height: strconv.FormatUint(record.Height, 10),
|
|
}
|
|
}
|
|
|
|
func getGQLBond(bondObj *bondtypes.Bond) (*Bond, error) {
|
|
// Nil record.
|
|
if bondObj == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
return &Bond{
|
|
ID: bondObj.Id,
|
|
Owner: bondObj.Owner,
|
|
Balance: getGQLCoins(bondObj.Balance),
|
|
}, nil
|
|
}
|
|
|
|
func getAuctionBid(bid *auctiontypes.Bid) *AuctionBid {
|
|
return &AuctionBid{
|
|
BidderAddress: bid.BidderAddress,
|
|
Status: bid.Status,
|
|
CommitHash: bid.CommitHash,
|
|
CommitTime: bid.GetCommitTime(),
|
|
RevealTime: bid.GetRevealTime(),
|
|
CommitFee: getGQLCoin(bid.CommitFee),
|
|
RevealFee: getGQLCoin(bid.RevealFee),
|
|
BidAmount: getGQLCoin(bid.BidAmount),
|
|
}
|
|
}
|
|
|
|
func GetGQLAuction(auction *auctiontypes.Auction, bids []*auctiontypes.Bid) (*Auction, error) {
|
|
if auction == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
winnerAddresses := make([]*string, len(auction.WinnerAddresses))
|
|
for i, winnerAddress := range auction.WinnerAddresses {
|
|
address := winnerAddress
|
|
winnerAddresses[i] = &address
|
|
}
|
|
|
|
numProviders := int(auction.NumProviders)
|
|
|
|
gqlAuction := Auction{
|
|
ID: auction.Id,
|
|
Status: auction.Status,
|
|
OwnerAddress: auction.OwnerAddress,
|
|
CreateTime: auction.GetCreateTime(),
|
|
CommitsEndTime: auction.GetCommitsEndTime(),
|
|
RevealsEndTime: auction.GetRevealsEndTime(),
|
|
CommitFee: getGQLCoin(auction.CommitFee),
|
|
RevealFee: getGQLCoin(auction.RevealFee),
|
|
MinimumBid: getGQLCoin(auction.MinimumBid),
|
|
WinnerAddresses: winnerAddresses,
|
|
WinnerBids: getGQLCoins(auction.WinningBids),
|
|
WinnerPrice: getGQLCoin(auction.WinningPrice),
|
|
MaxPrice: getGQLCoin(auction.MaxPrice),
|
|
Kind: auction.Kind,
|
|
NumProviders: &numProviders,
|
|
}
|
|
|
|
auctionBids := make([]*AuctionBid, len(bids))
|
|
for index, entry := range bids {
|
|
auctionBids[index] = getAuctionBid(entry)
|
|
}
|
|
|
|
gqlAuction.Bids = auctionBids
|
|
|
|
return &gqlAuction, nil
|
|
}
|
|
|
|
func toRPCValue(value *ValueInput) *registrytypes.QueryRecordsRequest_ValueInput {
|
|
var rpcval registrytypes.QueryRecordsRequest_ValueInput
|
|
|
|
switch {
|
|
case value == nil:
|
|
return nil
|
|
case value.Int != nil:
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Int{Int: int64(*value.Int)}
|
|
case value.Float != nil:
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Float{Float: *value.Float}
|
|
case value.String != nil:
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_String_{String_: *value.String}
|
|
case value.Boolean != nil:
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Boolean{Boolean: *value.Boolean}
|
|
case value.Link != nil:
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Link{Link: value.Link.String()}
|
|
case value.Array != nil:
|
|
var contents registrytypes.QueryRecordsRequest_ArrayInput
|
|
for _, val := range value.Array {
|
|
contents.Values = append(contents.Values, toRPCValue(val))
|
|
}
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Array{Array: &contents}
|
|
case value.Map != nil:
|
|
var contents registrytypes.QueryRecordsRequest_MapInput
|
|
for _, kv := range value.Map {
|
|
contents.Values[kv.Key] = toRPCValue(kv.Value)
|
|
}
|
|
rpcval.Value = ®istrytypes.QueryRecordsRequest_ValueInput_Map{Map: &contents}
|
|
}
|
|
return &rpcval
|
|
}
|
|
|
|
func toRPCAttributes(attrs []*KeyValueInput) []*registrytypes.QueryRecordsRequest_KeyValueInput {
|
|
kvPairs := []*registrytypes.QueryRecordsRequest_KeyValueInput{}
|
|
|
|
for _, value := range attrs {
|
|
parsedValue := toRPCValue(value.Value)
|
|
kvPair := ®istrytypes.QueryRecordsRequest_KeyValueInput{
|
|
Key: value.Key,
|
|
Value: parsedValue,
|
|
}
|
|
kvPairs = append(kvPairs, kvPair)
|
|
}
|
|
|
|
return kvPairs
|
|
}
|
|
|
|
func getAuthorityRecord(nameAuthority registrytypes.NameAuthority, auctionQueryClient auctiontypes.QueryClient) (*AuthorityRecord, error) {
|
|
gqlNameAuthorityRecord, err := GetGQLNameAuthorityRecord(&nameAuthority)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if nameAuthority.AuctionId != "" {
|
|
auctionResp, err := auctionQueryClient.GetAuction(context.Background(), &auctiontypes.QueryGetAuctionRequest{Id: nameAuthority.GetAuctionId()})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
bidsResp, err := auctionQueryClient.GetBids(context.Background(), &auctiontypes.QueryGetBidsRequest{AuctionId: nameAuthority.GetAuctionId()})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
gqlAuctionRecord, err := GetGQLAuction(auctionResp.GetAuction(), bidsResp.GetBids())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
gqlNameAuthorityRecord.Auction = gqlAuctionRecord
|
|
}
|
|
|
|
return gqlNameAuthorityRecord, nil
|
|
}
|