2023-02-21 10:05:16 +01:00
|
|
|
package opensea
|
2021-08-20 21:53:24 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2021-09-20 18:24:07 +02:00
|
|
|
"errors"
|
2021-08-20 21:53:24 +02:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2021-08-30 09:50:18 +02:00
|
|
|
"strconv"
|
2022-08-03 09:42:56 +02:00
|
|
|
"strings"
|
2023-01-17 10:56:16 +01:00
|
|
|
"sync"
|
2021-08-20 21:53:24 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
)
|
|
|
|
|
|
|
|
const AssetLimit = 50
|
|
|
|
const CollectionLimit = 300
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
var OpenseaClientInstances = make(map[uint64]*Client)
|
2023-01-17 10:56:16 +01:00
|
|
|
|
2021-09-20 18:24:07 +02:00
|
|
|
var BaseURLs = map[uint64]string{
|
|
|
|
1: "https://api.opensea.io/api/v1",
|
|
|
|
4: "https://rinkeby-api.opensea.io/api/v1",
|
2022-10-12 09:59:38 +02:00
|
|
|
5: "https://testnets-api.opensea.io/api/v1",
|
2021-09-20 18:24:07 +02:00
|
|
|
}
|
|
|
|
|
2021-08-30 09:50:18 +02:00
|
|
|
type TraitValue string
|
|
|
|
|
|
|
|
func (st *TraitValue) UnmarshalJSON(b []byte) error {
|
|
|
|
var item interface{}
|
|
|
|
if err := json.Unmarshal(b, &item); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-09-17 10:01:35 +02:00
|
|
|
|
2021-08-30 09:50:18 +02:00
|
|
|
switch v := item.(type) {
|
2021-09-17 10:01:35 +02:00
|
|
|
case float64:
|
|
|
|
*st = TraitValue(strconv.FormatFloat(v, 'f', 2, 64))
|
2021-08-30 09:50:18 +02:00
|
|
|
case int:
|
|
|
|
*st = TraitValue(strconv.Itoa(v))
|
|
|
|
case string:
|
|
|
|
*st = TraitValue(v)
|
|
|
|
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type AssetContainer struct {
|
|
|
|
Assets []Asset `json:"assets"`
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type AssetCollection struct {
|
2021-08-20 21:53:24 +02:00
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type Contract struct {
|
2021-08-20 21:53:24 +02:00
|
|
|
Address string `json:"address"`
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type Trait struct {
|
2021-08-30 09:50:18 +02:00
|
|
|
TraitType string `json:"trait_type"`
|
|
|
|
Value TraitValue `json:"value"`
|
|
|
|
DisplayType string `json:"display_type"`
|
2021-09-17 10:01:35 +02:00
|
|
|
MaxValue string `json:"max_value"`
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type PaymentToken struct {
|
2021-09-17 10:01:35 +02:00
|
|
|
ID int `json:"id"`
|
|
|
|
Symbol string `json:"symbol"`
|
|
|
|
Address string `json:"address"`
|
|
|
|
ImageURL string `json:"image_url"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Decimals int `json:"decimals"`
|
|
|
|
EthPrice string `json:"eth_price"`
|
|
|
|
UsdPrice string `json:"usd_price"`
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type LastSale struct {
|
|
|
|
PaymentToken PaymentToken `json:"payment_token"`
|
2021-08-30 09:50:18 +02:00
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type SellOrder struct {
|
2021-09-17 10:01:35 +02:00
|
|
|
CurrentPrice string `json:"current_price"`
|
|
|
|
}
|
2023-02-21 10:05:16 +01:00
|
|
|
type Asset struct {
|
|
|
|
ID int `json:"id"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Description string `json:"description"`
|
|
|
|
Permalink string `json:"permalink"`
|
|
|
|
ImageThumbnailURL string `json:"image_thumbnail_url"`
|
|
|
|
ImageURL string `json:"image_url"`
|
|
|
|
Contract Contract `json:"asset_contract"`
|
|
|
|
Collection AssetCollection `json:"collection"`
|
|
|
|
Traits []Trait `json:"traits"`
|
|
|
|
LastSale LastSale `json:"last_sale"`
|
|
|
|
SellOrders []SellOrder `json:"sell_orders"`
|
|
|
|
BackgroundColor string `json:"background_color"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type CollectionTrait struct {
|
2021-09-17 10:01:35 +02:00
|
|
|
Min float64 `json:"min"`
|
|
|
|
Max float64 `json:"max"`
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type Collection struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Slug string `json:"slug"`
|
|
|
|
ImageURL string `json:"image_url"`
|
|
|
|
OwnedAssetCount int `json:"owned_asset_count"`
|
|
|
|
Traits map[string]CollectionTrait `json:"traits"`
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type Client struct {
|
2023-01-17 10:56:16 +01:00
|
|
|
client *http.Client
|
|
|
|
url string
|
|
|
|
apiKey string
|
|
|
|
IsConnected bool
|
2023-02-23 14:55:57 +01:00
|
|
|
LastCheckedAt int64
|
2023-01-17 10:56:16 +01:00
|
|
|
IsConnectedLock sync.RWMutex
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// new opensea client.
|
2023-02-21 10:05:16 +01:00
|
|
|
func NewOpenseaClient(chainID uint64, apiKey string) (*Client, error) {
|
2023-01-17 10:56:16 +01:00
|
|
|
if client, ok := OpenseaClientInstances[chainID]; ok {
|
|
|
|
if client.apiKey == apiKey {
|
|
|
|
return client, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-20 21:53:24 +02:00
|
|
|
client := &http.Client{
|
|
|
|
Timeout: time.Second * 5,
|
|
|
|
}
|
2021-09-20 18:24:07 +02:00
|
|
|
if url, ok := BaseURLs[chainID]; ok {
|
2023-02-21 10:05:16 +01:00
|
|
|
openseaClient := &Client{client: client, url: url, apiKey: apiKey, IsConnected: true, LastCheckedAt: time.Now().Unix()}
|
2023-01-17 10:56:16 +01:00
|
|
|
OpenseaClientInstances[chainID] = openseaClient
|
|
|
|
return openseaClient, nil
|
2021-09-20 18:24:07 +02:00
|
|
|
}
|
2021-08-20 21:53:24 +02:00
|
|
|
|
2021-09-20 18:24:07 +02:00
|
|
|
return nil, errors.New("ChainID not supported")
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
func (o *Client) FetchAllCollectionsByOwner(owner common.Address) ([]Collection, error) {
|
2021-08-20 21:53:24 +02:00
|
|
|
offset := 0
|
2023-02-21 10:05:16 +01:00
|
|
|
var collections []Collection
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnectedLock.Lock()
|
|
|
|
defer o.IsConnectedLock.Unlock()
|
2023-02-23 14:55:57 +01:00
|
|
|
o.LastCheckedAt = time.Now().Unix()
|
2021-08-20 21:53:24 +02:00
|
|
|
for {
|
|
|
|
url := fmt.Sprintf("%s/collections?asset_owner=%s&offset=%d&limit=%d", o.url, owner, offset, CollectionLimit)
|
|
|
|
body, err := o.doOpenseaRequest(url)
|
|
|
|
if err != nil {
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnected = false
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
var tmp []Collection
|
2021-08-20 21:53:24 +02:00
|
|
|
err = json.Unmarshal(body, &tmp)
|
|
|
|
if err != nil {
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnected = false
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
collections = append(collections, tmp...)
|
|
|
|
|
|
|
|
if len(tmp) < CollectionLimit {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnected = true
|
2021-08-20 21:53:24 +02:00
|
|
|
return collections, nil
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
func (o *Client) FetchAllAssetsByOwnerAndCollection(owner common.Address, collectionSlug string, limit int) ([]Asset, error) {
|
2021-08-20 21:53:24 +02:00
|
|
|
offset := 0
|
2023-02-21 10:05:16 +01:00
|
|
|
var assets []Asset
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnectedLock.Lock()
|
|
|
|
defer o.IsConnectedLock.Unlock()
|
2023-02-23 14:55:57 +01:00
|
|
|
o.LastCheckedAt = time.Now().Unix()
|
2021-08-20 21:53:24 +02:00
|
|
|
for {
|
|
|
|
url := fmt.Sprintf("%s/assets?owner=%s&collection=%s&offset=%d&limit=%d", o.url, owner, collectionSlug, offset, AssetLimit)
|
|
|
|
body, err := o.doOpenseaRequest(url)
|
|
|
|
if err != nil {
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnected = false
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
container := AssetContainer{}
|
2021-08-20 21:53:24 +02:00
|
|
|
err = json.Unmarshal(body, &container)
|
|
|
|
if err != nil {
|
2023-01-17 10:56:16 +01:00
|
|
|
o.IsConnected = false
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-08-03 09:42:56 +02:00
|
|
|
for _, asset := range container.Assets {
|
|
|
|
for i := range asset.Traits {
|
2022-11-29 09:28:37 -03:00
|
|
|
asset.Traits[i].TraitType = strings.Replace(asset.Traits[i].TraitType, "_", " ", 1)
|
2022-08-03 09:42:56 +02:00
|
|
|
asset.Traits[i].Value = TraitValue(strings.Title(string(asset.Traits[i].Value)))
|
|
|
|
}
|
|
|
|
assets = append(assets, asset)
|
|
|
|
}
|
2021-08-20 21:53:24 +02:00
|
|
|
|
|
|
|
if len(container.Assets) < AssetLimit {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(assets) >= limit {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2023-01-17 10:56:16 +01:00
|
|
|
|
|
|
|
o.IsConnected = true
|
2021-08-20 21:53:24 +02:00
|
|
|
return assets, nil
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
func (o *Client) doOpenseaRequest(url string) ([]byte, error) {
|
2021-08-20 21:53:24 +02:00
|
|
|
req, err := http.NewRequest(http.MethodGet, url, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-17 10:56:16 +01:00
|
|
|
|
2021-08-20 21:53:24 +02:00
|
|
|
req.Header.Set("Content-Type", "application/json")
|
2022-01-18 10:53:34 +01:00
|
|
|
req.Header.Set("User-Agent", "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:96.0) Gecko/20100101 Firefox/96.0")
|
2022-02-16 10:22:19 +01:00
|
|
|
req.Header.Set("X-API-KEY", o.apiKey)
|
2021-08-20 21:53:24 +02:00
|
|
|
|
|
|
|
resp, err := o.client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if err := resp.Body.Close(); err != nil {
|
|
|
|
log.Error("failed to close opensea request body", "err", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
return body, err
|
|
|
|
}
|