2023-02-21 10:05:16 +01:00
|
|
|
package opensea
|
2021-08-20 21:53:24 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2023-02-28 00:10:38 -03:00
|
|
|
"net/url"
|
2021-08-30 09:50:18 +02:00
|
|
|
"strconv"
|
2022-08-03 09:42:56 +02:00
|
|
|
"strings"
|
2021-08-20 21:53:24 +02:00
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2023-10-03 15:53:36 -03:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
2023-02-28 00:10:38 -03:00
|
|
|
|
2023-04-18 11:33:59 -03:00
|
|
|
walletCommon "github.com/status-im/status-go/services/wallet/common"
|
2023-07-10 06:02:17 -03:00
|
|
|
"github.com/status-im/status-go/services/wallet/connection"
|
2023-03-21 10:52:14 -03:00
|
|
|
"github.com/status-im/status-go/services/wallet/thirdparty"
|
2021-08-20 21:53:24 +02:00
|
|
|
)
|
|
|
|
|
2023-03-06 13:15:48 -03:00
|
|
|
const AssetLimit = 200
|
2021-08-20 21:53:24 +02:00
|
|
|
const CollectionLimit = 300
|
|
|
|
|
2023-07-12 14:43:22 -03:00
|
|
|
const ChainIDRequiringAPIKey = walletCommon.EthereumMainnet
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func getBaseURL(chainID walletCommon.ChainID) (string, error) {
|
2023-07-18 12:01:53 -03:00
|
|
|
// v1 Endpoints only support L1 chain
|
|
|
|
switch uint64(chainID) {
|
2023-07-12 14:43:22 -03:00
|
|
|
case walletCommon.EthereumMainnet:
|
2023-03-27 15:50:19 -03:00
|
|
|
return "https://api.opensea.io/api/v1", nil
|
2023-08-17 15:10:13 -03:00
|
|
|
case walletCommon.EthereumSepolia:
|
2023-03-27 15:50:19 -03:00
|
|
|
return "https://testnets-api.opensea.io/api/v1", nil
|
|
|
|
}
|
2023-01-17 10:56:16 +01:00
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
return "", thirdparty.ErrChainIDNotSupported
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Client) ID() string {
|
2023-07-31 20:34:53 -03:00
|
|
|
return OpenseaV1ID
|
2023-07-31 16:41:14 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Client) IsChainSupported(chainID walletCommon.ChainID) bool {
|
|
|
|
_, err := getBaseURL(chainID)
|
|
|
|
return err == nil
|
2021-09-20 18:24:07 +02:00
|
|
|
}
|
|
|
|
|
2023-09-22 10:18:42 -03:00
|
|
|
func (o *Client) IsConnected() bool {
|
|
|
|
return o.connectionStatus.IsConnected()
|
|
|
|
}
|
|
|
|
|
2023-07-18 12:01:53 -03:00
|
|
|
func getURL(chainID walletCommon.ChainID, path string) (string, error) {
|
2023-07-31 16:41:14 -03:00
|
|
|
baseURL, err := getBaseURL(chainID)
|
2023-07-18 12:01:53 -03:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%s/%s", baseURL, path), nil
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:05:16 +01:00
|
|
|
type Client struct {
|
2023-07-10 06:02:17 -03:00
|
|
|
client *HTTPClient
|
|
|
|
apiKey string
|
|
|
|
connectionStatus *connection.Status
|
2023-07-18 12:01:53 -03:00
|
|
|
urlGetter urlGetter
|
2021-08-20 21:53:24 +02:00
|
|
|
}
|
|
|
|
|
2023-08-17 15:10:13 -03:00
|
|
|
// new opensea v1 client.
|
2023-09-22 10:18:42 -03:00
|
|
|
func NewClient(apiKey string, httpClient *HTTPClient) *Client {
|
2023-10-03 15:53:36 -03:00
|
|
|
if apiKey == "" {
|
|
|
|
log.Warn("OpenseaV1 API key not available")
|
|
|
|
}
|
|
|
|
|
2023-07-10 06:02:17 -03:00
|
|
|
return &Client{
|
2023-08-17 15:10:13 -03:00
|
|
|
client: httpClient,
|
2023-07-10 06:02:17 -03:00
|
|
|
apiKey: apiKey,
|
2023-09-22 10:18:42 -03:00
|
|
|
connectionStatus: connection.NewStatus(),
|
2023-07-18 12:01:53 -03:00
|
|
|
urlGetter: getURL,
|
2023-03-27 15:50:19 -03:00
|
|
|
}
|
2023-07-10 06:02:17 -03:00
|
|
|
}
|
2023-03-27 15:50:19 -03:00
|
|
|
|
2023-07-18 12:01:53 -03:00
|
|
|
func (o *Client) FetchAllCollectionsByOwner(chainID walletCommon.ChainID, owner common.Address) ([]OwnedCollection, error) {
|
2023-07-10 06:02:17 -03:00
|
|
|
offset := 0
|
|
|
|
var collections []OwnedCollection
|
2023-01-17 10:56:16 +01:00
|
|
|
|
2021-08-20 21:53:24 +02:00
|
|
|
for {
|
2023-07-18 12:01:53 -03:00
|
|
|
path := fmt.Sprintf("collections?asset_owner=%s&offset=%d&limit=%d", owner, offset, CollectionLimit)
|
|
|
|
url, err := o.urlGetter(chainID, path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-03-27 15:50:19 -03:00
|
|
|
body, err := o.client.doGetRequest(url, o.apiKey)
|
2021-08-20 21:53:24 +02:00
|
|
|
if err != nil {
|
2023-07-10 06:02:17 -03:00
|
|
|
o.connectionStatus.SetIsConnected(false)
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
2023-07-10 06:02:17 -03:00
|
|
|
o.connectionStatus.SetIsConnected(true)
|
2021-08-20 21:53:24 +02:00
|
|
|
|
2023-03-20 16:53:39 +01:00
|
|
|
// if Json is not returned there must be an error
|
|
|
|
if !json.Valid(body) {
|
|
|
|
return nil, fmt.Errorf("invalid json: %s", string(body))
|
|
|
|
}
|
|
|
|
|
2023-02-28 00:10:38 -03:00
|
|
|
var tmp []OwnedCollection
|
2021-08-20 21:53:24 +02:00
|
|
|
err = json.Unmarshal(body, &tmp)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
collections = append(collections, tmp...)
|
|
|
|
|
|
|
|
if len(tmp) < CollectionLimit {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return collections, nil
|
|
|
|
}
|
|
|
|
|
2023-08-03 09:24:23 -03:00
|
|
|
func (o *Client) FetchCollectionsDataByContractID(ids []thirdparty.ContractID) ([]thirdparty.CollectionData, error) {
|
|
|
|
ret := make([]thirdparty.CollectionData, 0, len(ids))
|
|
|
|
|
|
|
|
for _, id := range ids {
|
|
|
|
path := fmt.Sprintf("asset_contract/%s", id.Address.String())
|
|
|
|
url, err := o.urlGetter(id.ChainID, path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
body, err := o.client.doGetRequest(url, o.apiKey)
|
|
|
|
if err != nil {
|
|
|
|
o.connectionStatus.SetIsConnected(false)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
o.connectionStatus.SetIsConnected(true)
|
|
|
|
|
|
|
|
// if Json is not returned there must be an error
|
|
|
|
if !json.Valid(body) {
|
|
|
|
return nil, fmt.Errorf("invalid json: %s", string(body))
|
|
|
|
}
|
|
|
|
|
|
|
|
var tmp AssetContract
|
|
|
|
err = json.Unmarshal(body, &tmp)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, tmp.Collection.toCollectionData(id))
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func (o *Client) FetchAllAssetsByOwnerAndCollection(chainID walletCommon.ChainID, owner common.Address, collectionSlug string, cursor string, limit int) (*thirdparty.FullCollectibleDataContainer, error) {
|
2023-02-28 00:10:38 -03:00
|
|
|
queryParams := url.Values{
|
|
|
|
"owner": {owner.String()},
|
|
|
|
"collection": {collectionSlug},
|
|
|
|
}
|
|
|
|
|
2023-03-06 13:15:48 -03:00
|
|
|
if len(cursor) > 0 {
|
|
|
|
queryParams["cursor"] = []string{cursor}
|
|
|
|
}
|
|
|
|
|
2023-07-10 06:02:17 -03:00
|
|
|
return o.fetchAssets(chainID, queryParams, limit)
|
2023-03-06 13:15:48 -03:00
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func (o *Client) FetchAllAssetsByOwnerAndContractAddress(chainID walletCommon.ChainID, owner common.Address, contractAddresses []common.Address, cursor string, limit int) (*thirdparty.FullCollectibleDataContainer, error) {
|
2023-03-17 18:20:31 -03:00
|
|
|
queryParams := url.Values{
|
|
|
|
"owner": {owner.String()},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, contractAddress := range contractAddresses {
|
|
|
|
queryParams.Add("asset_contract_addresses", contractAddress.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cursor) > 0 {
|
|
|
|
queryParams["cursor"] = []string{cursor}
|
|
|
|
}
|
|
|
|
|
2023-07-10 06:02:17 -03:00
|
|
|
return o.fetchAssets(chainID, queryParams, limit)
|
2023-03-17 18:20:31 -03:00
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func (o *Client) FetchAllAssetsByOwner(chainID walletCommon.ChainID, owner common.Address, cursor string, limit int) (*thirdparty.FullCollectibleDataContainer, error) {
|
2023-03-06 13:15:48 -03:00
|
|
|
queryParams := url.Values{
|
|
|
|
"owner": {owner.String()},
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cursor) > 0 {
|
|
|
|
queryParams["cursor"] = []string{cursor}
|
|
|
|
}
|
|
|
|
|
2023-07-10 06:02:17 -03:00
|
|
|
return o.fetchAssets(chainID, queryParams, limit)
|
2023-02-28 00:10:38 -03:00
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func (o *Client) FetchAssetsByCollectibleUniqueID(uniqueIDs []thirdparty.CollectibleUniqueID) ([]thirdparty.FullCollectibleData, error) {
|
2023-02-28 00:10:38 -03:00
|
|
|
queryParams := url.Values{}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
ret := make([]thirdparty.FullCollectibleData, 0, len(uniqueIDs))
|
2023-07-18 12:01:53 -03:00
|
|
|
|
|
|
|
idsPerChainID := thirdparty.GroupCollectibleUIDsByChainID(uniqueIDs)
|
|
|
|
for chainID, ids := range idsPerChainID {
|
|
|
|
for _, id := range ids {
|
|
|
|
queryParams.Add("token_ids", id.TokenID.String())
|
2023-07-31 16:41:14 -03:00
|
|
|
queryParams.Add("asset_contract_addresses", id.ContractID.Address.String())
|
2023-07-18 12:01:53 -03:00
|
|
|
}
|
|
|
|
|
2023-07-31 20:34:53 -03:00
|
|
|
data, err := o.fetchAssets(chainID, queryParams, thirdparty.FetchNoLimit)
|
2023-07-18 12:01:53 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
ret = append(ret, data.Items...)
|
2023-02-28 00:10:38 -03:00
|
|
|
}
|
|
|
|
|
2023-07-18 12:01:53 -03:00
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
func (o *Client) fetchAssets(chainID walletCommon.ChainID, queryParams url.Values, limit int) (*thirdparty.FullCollectibleDataContainer, error) {
|
|
|
|
assets := new(thirdparty.FullCollectibleDataContainer)
|
2023-07-18 12:01:53 -03:00
|
|
|
|
|
|
|
if len(queryParams["cursor"]) > 0 {
|
|
|
|
assets.PreviousCursor = queryParams["cursor"][0]
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpLimit := AssetLimit
|
2023-07-31 20:34:53 -03:00
|
|
|
if limit > thirdparty.FetchNoLimit && limit < tmpLimit {
|
2023-07-18 12:01:53 -03:00
|
|
|
tmpLimit = limit
|
|
|
|
}
|
|
|
|
|
|
|
|
queryParams["limit"] = []string{strconv.Itoa(tmpLimit)}
|
|
|
|
for {
|
|
|
|
path := "assets?" + queryParams.Encode()
|
|
|
|
url, err := o.urlGetter(chainID, path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
body, err := o.client.doGetRequest(url, o.apiKey)
|
|
|
|
if err != nil {
|
|
|
|
o.connectionStatus.SetIsConnected(false)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
o.connectionStatus.SetIsConnected(true)
|
|
|
|
|
|
|
|
// if Json is not returned there must be an error
|
|
|
|
if !json.Valid(body) {
|
|
|
|
return nil, fmt.Errorf("invalid json: %s", string(body))
|
|
|
|
}
|
|
|
|
|
|
|
|
container := AssetContainer{}
|
|
|
|
err = json.Unmarshal(body, &container)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, asset := range container.Assets {
|
2023-07-31 16:41:14 -03:00
|
|
|
assets.Items = append(assets.Items, asset.toCommon())
|
2023-07-18 12:01:53 -03:00
|
|
|
}
|
|
|
|
assets.NextCursor = container.NextCursor
|
|
|
|
|
|
|
|
if len(assets.NextCursor) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
queryParams["cursor"] = []string{assets.NextCursor}
|
|
|
|
|
2023-07-31 20:34:53 -03:00
|
|
|
if limit > thirdparty.FetchNoLimit && len(assets.Items) >= limit {
|
2023-07-18 12:01:53 -03:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return assets, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only here for compatibility with mobile app, to be removed
|
|
|
|
func (o *Client) FetchAllOpenseaAssetsByOwnerAndCollection(chainID walletCommon.ChainID, owner common.Address, collectionSlug string, cursor string, limit int) (*AssetContainer, error) {
|
|
|
|
queryParams := url.Values{
|
|
|
|
"owner": {owner.String()},
|
|
|
|
"collection": {collectionSlug},
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cursor) > 0 {
|
|
|
|
queryParams["cursor"] = []string{cursor}
|
|
|
|
}
|
|
|
|
|
|
|
|
return o.fetchOpenseaAssets(chainID, queryParams, limit)
|
2023-02-28 00:10:38 -03:00
|
|
|
}
|
|
|
|
|
2023-07-18 12:01:53 -03:00
|
|
|
func (o *Client) fetchOpenseaAssets(chainID walletCommon.ChainID, queryParams url.Values, limit int) (*AssetContainer, error) {
|
2023-03-06 13:15:48 -03:00
|
|
|
assets := new(AssetContainer)
|
|
|
|
|
|
|
|
if len(queryParams["cursor"]) > 0 {
|
|
|
|
assets.PreviousCursor = queryParams["cursor"][0]
|
|
|
|
}
|
|
|
|
|
2023-03-17 18:20:31 -03:00
|
|
|
tmpLimit := AssetLimit
|
|
|
|
if limit > 0 && limit < tmpLimit {
|
2023-06-15 10:51:55 -03:00
|
|
|
tmpLimit = limit
|
2023-03-06 13:15:48 -03:00
|
|
|
}
|
2023-02-28 00:10:38 -03:00
|
|
|
|
2023-07-31 16:41:14 -03:00
|
|
|
baseURL, err := getBaseURL(chainID)
|
2023-07-10 06:02:17 -03:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-03-06 13:15:48 -03:00
|
|
|
queryParams["limit"] = []string{strconv.Itoa(tmpLimit)}
|
2021-08-20 21:53:24 +02:00
|
|
|
for {
|
2023-07-10 06:02:17 -03:00
|
|
|
url := baseURL + "/assets?" + queryParams.Encode()
|
2023-02-28 00:10:38 -03:00
|
|
|
|
2023-03-27 15:50:19 -03:00
|
|
|
body, err := o.client.doGetRequest(url, o.apiKey)
|
2021-08-20 21:53:24 +02:00
|
|
|
if err != nil {
|
2023-07-10 06:02:17 -03:00
|
|
|
o.connectionStatus.SetIsConnected(false)
|
2021-08-20 21:53:24 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
2023-07-10 06:02:17 -03:00
|
|
|
o.connectionStatus.SetIsConnected(true)
|
2021-08-20 21:53:24 +02:00
|
|
|
|
2023-03-20 16:53:39 +01:00
|
|
|
// if Json is not returned there must be an error
|
|
|
|
if !json.Valid(body) {
|
|
|
|
return nil, fmt.Errorf("invalid json: %s", string(body))
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
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)))
|
|
|
|
}
|
2023-03-06 13:15:48 -03:00
|
|
|
assets.Assets = append(assets.Assets, asset)
|
2022-08-03 09:42:56 +02:00
|
|
|
}
|
2023-03-06 13:15:48 -03:00
|
|
|
assets.NextCursor = container.NextCursor
|
2021-08-20 21:53:24 +02:00
|
|
|
|
2023-03-06 13:15:48 -03:00
|
|
|
if len(assets.NextCursor) == 0 {
|
2023-02-28 00:10:38 -03:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2023-03-06 13:15:48 -03:00
|
|
|
queryParams["cursor"] = []string{assets.NextCursor}
|
2023-02-28 00:10:38 -03:00
|
|
|
|
2023-03-17 18:20:31 -03:00
|
|
|
if limit > 0 && len(assets.Assets) >= limit {
|
2021-08-20 21:53:24 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2023-01-17 10:56:16 +01:00
|
|
|
|
2021-08-20 21:53:24 +02:00
|
|
|
return assets, nil
|
|
|
|
}
|