-
Notifications
You must be signed in to change notification settings - Fork 3
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
feat: metadata api #44
Changes from 11 commits
e025ecd
94c109e
fa69de9
73507fc
a5f9763
ee657c0
76b38df
ce8d7be
8484d2a
1ff0fac
81b68b5
5906637
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,32 @@ | ||
from dataclasses import asdict, dataclass | ||
from enum import Enum | ||
|
||
|
||
class MetadataType(str, Enum): | ||
TOKEN = 'TOKEN' | ||
TRANSACTION = 'TRANSACTION' | ||
|
||
|
||
@dataclass | ||
class Metadata: | ||
id: str | ||
|
||
@property | ||
def type(self) -> MetadataType: | ||
raise NotImplementedError | ||
|
||
@property | ||
def data(self): | ||
raise NotImplementedError | ||
|
||
@classmethod | ||
def from_dict(cls, dikt: dict) -> 'Metadata': | ||
raise NotImplementedError | ||
|
||
def to_dict(self) -> dict: | ||
""" Convert a Metadata instance into dict | ||
|
||
:return: Dict representations of Metadata | ||
:rtype: dict | ||
""" | ||
return asdict(self) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
from dataclasses import dataclass | ||
from enum import Enum | ||
from typing import Optional | ||
|
||
from domain.metadata.metadata import Metadata, MetadataType | ||
|
||
|
||
class TokenNFTMediaType(str, Enum): | ||
VIDEO = 'VIDEO' | ||
IMAGE = 'IMAGE' | ||
AUDIO = 'AUDIO' | ||
|
||
|
||
@dataclass | ||
class TokenNFTMedia: | ||
"""Data of token NFT | ||
|
||
:param type: NFT Type. image or video. | ||
:type type: :py:class:`domain.metadata.token_metdata.TokenNFTMediaType` | ||
|
||
:param file: Media file of the NFT | ||
:type file: str | ||
|
||
:param loop: If media will play in loop or not. Works for audio and video as well | ||
:type loop: Optional[bool] | ||
|
||
:param autoplay: If media will play automatically or not. Works for audio and video as well | ||
:type autoplay: Optional[bool] | ||
""" | ||
type: TokenNFTMediaType | ||
file: str | ||
loop: Optional[bool] = False | ||
autoplay: Optional[bool] = False | ||
|
||
|
||
@dataclass | ||
class MetaToken: | ||
"""Metadata of a token | ||
|
||
:param id: Token unique id | ||
:type id: str | ||
|
||
:param verified: If token is verified or not. None and False are equivalent | ||
:type verified: Optional[bool] | ||
|
||
:param banned: If token is banned or not. None and False are equivalent | ||
:type banned: Optional[bool] | ||
|
||
:param reason: Bannishment reason | ||
:type reason: Optional[str] | ||
|
||
:param nft: If token is a nft or not. None and False are equivalent | ||
:type nft: Optional[bool] | ||
|
||
:param nft_media: NFT media data, if any. | ||
:type nft_media: Optional[:py:class:`domain.metadata.token_metdata.TokenNFTMedia`] | ||
""" | ||
id: str | ||
verified: Optional[bool] = False | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Just to let the json have only the fields that matter. Docstring updated. |
||
banned: Optional[bool] = False | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Just to let the json have only the fields that matter. Docstring updated. |
||
reason: Optional[str] = '' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This property only make sense for banned tokens |
||
nft: Optional[bool] = False | ||
nft_media: Optional[TokenNFTMedia] = None | ||
|
||
|
||
@dataclass | ||
class TokenMetadata(Metadata): | ||
data: Optional[MetaToken] = None | ||
type: MetadataType = MetadataType.TOKEN | ||
|
||
@classmethod | ||
def from_dict(cls, dikt: dict) -> 'TokenMetadata': | ||
""" Creates a new TokenMetadata instance from a given dict (inverse operation of `to_dict`) | ||
|
||
:param dikt: Dict with TokenMetadata structure and data | ||
:type dikt: dict | ||
|
||
:return: The new instance | ||
:rtype: :py:class:`domain.metadata.token_metdata.TokenMetadata` | ||
""" | ||
data = dikt.get('data', {}) | ||
if data.get('nft_media'): | ||
data['nft_media'] = TokenNFTMedia( | ||
TokenNFTMediaType(data['nft_media']['type'].upper()), | ||
data['nft_media']['file'], | ||
data['nft_media'].get('loop', None) | ||
) | ||
|
||
return cls( | ||
id=dikt['id'], | ||
data=MetaToken(**data) | ||
) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
from dataclasses import dataclass | ||
from typing import Optional | ||
|
||
from domain.metadata.metadata import Metadata, MetadataType | ||
|
||
|
||
@dataclass | ||
class MetaTransaction: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If we add any new property in transaction metadata we need to update the explorer service? Imagine we want to mark a tx as banned or as important (don't know why but just imagining possible situations). How do we handle this? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, and i think that is the right thing to do. Otherwise we could broke things in explorer or have other unpredictable problems. |
||
id: str | ||
context: Optional[str] | ||
genesis: Optional[bool] | ||
|
||
|
||
@dataclass | ||
class TransactionMetadata(Metadata): | ||
data: Optional[MetaTransaction] = None | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why are we using two classes? I feel it would be simper to use one class with all fields there. Well, we can group in subclasses for specific cases if there are too many fields. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We have only specific cases by now. We can use |
||
type: MetadataType = MetadataType.TRANSACTION | ||
|
||
@classmethod | ||
def from_dict(cls, dikt: dict) -> 'TransactionMetadata': | ||
""" Creates a new TransactionMetadata instance from a given dict (inverse operation of `to_dict`) | ||
|
||
:param dikt: Dict with TransactionMetadata structure and data | ||
:type dikt: dict | ||
|
||
:return: The new instance | ||
:rtype: :py:class:`domain.metadata.token_metdata.TransactionMetadata` | ||
""" | ||
|
||
return cls( | ||
id=dikt['id'], | ||
data=MetaTransaction(**dikt['data']) | ||
) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,80 @@ | ||
import json | ||
from typing import Optional | ||
|
||
from common.configuration import METADATA_BUCKET | ||
from domain.metadata.metadata import MetadataType | ||
from domain.metadata.token_metadata import TokenMetadata | ||
from domain.metadata.transaction_metadata import TransactionMetadata | ||
from gateways.clients.s3_client import S3Client | ||
|
||
TYPE_TO_FOLDER = { | ||
MetadataType.TRANSACTION: 'transaction', | ||
MetadataType.TOKEN: 'token' | ||
} | ||
|
||
|
||
class MetadataGateway: | ||
|
||
def __init__(self, s3_client: Optional[S3Client] = None) -> None: | ||
self.s3_client = s3_client or S3Client() | ||
|
||
def get_transaction_metadata(self, id: str) -> Optional[TransactionMetadata]: | ||
"""Retrieve transaction metadata from a json file stored in s3 | ||
|
||
:param id: transaction id | ||
:type id: str | ||
:raises Exception: The name of the bucket used to store the jsons must be on config | ||
:return: TransactionMetadata object or None if not found | ||
:rtype: TokenMetadata | None | ||
""" | ||
folder = TYPE_TO_FOLDER[MetadataType.TRANSACTION] | ||
transaction_metadata = self._get_metadata(f"{folder}/{id}.json") | ||
|
||
if transaction_metadata is None: | ||
return None | ||
|
||
return TransactionMetadata.from_dict(transaction_metadata) | ||
|
||
def get_token_metadata(self, id: str) -> Optional[TokenMetadata]: | ||
pedroferreira1 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
"""Retrieve token metadata from a json file stored in s3 | ||
|
||
:param id: token id | ||
:type id: str | ||
|
||
:return: TokenMetadata object or None if not found | ||
:rtype: TokenMetadata | None | ||
""" | ||
folder = TYPE_TO_FOLDER[MetadataType.TOKEN] | ||
token_metadata = self._get_metadata(f"{folder}/{id}.json") | ||
|
||
if token_metadata is None: | ||
return None | ||
|
||
return TokenMetadata.from_dict(token_metadata) | ||
|
||
def _get_metadata(self, s3_object_name: str) -> Optional[dict]: | ||
"""Retrive metadata from file in s3 | ||
|
||
:param s3_object_name: name of object | ||
:type s3_object_name: str | ||
|
||
:raises Exception: The name of the bucket used to store the jsons must be on config | ||
|
||
:return: file content as dict | ||
:rtype: Optional[dict] | ||
""" | ||
metadata_bucket = self._metadata_bucket() | ||
|
||
raw_metadata = self.s3_client.load_file(metadata_bucket, s3_object_name) | ||
if raw_metadata is None: | ||
return None | ||
|
||
return json.loads(raw_metadata) | ||
pedroferreira1 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
def _metadata_bucket(self) -> str: | ||
metadata_bucket = METADATA_BUCKET | ||
|
||
if metadata_bucket is None: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Isn't it better to validate all configuration in the constructor? So we can assume everything is right when we get to this point? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think so as we can have methods that don't need this constant. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. All methods need this constant for now, right? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @pedroferreira1 yes, but i still feel it's not good to check in constructor. We don't have this var in test env, for instance and i don't think we must to. Of course for some tests we have to mock it, but some tests just don't care about it and i think it doesn't make sense to mock a var only for the class to not break. I know it's just test but i think it's a good indication why maybe it's not so good to check this on constructor. |
||
raise Exception('No bucket name in config') | ||
|
||
return metadata_bucket |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,48 +1,19 @@ | ||
import json | ||
from typing import Union | ||
|
||
from common.configuration import TOKEN_METADATA_BUCKET | ||
from domain.tx.token import Token, TokenMetadata | ||
from domain.tx.token import Token | ||
from gateways.clients.hathor_core_client import TOKEN_ENDPOINT, HathorCoreClient | ||
from gateways.clients.responses.hathor_core.hathor_core_token_response import HathorCoreTokenResponse | ||
from gateways.clients.s3_client import S3Client | ||
|
||
|
||
class TokenGateway: | ||
"""Gateway for Token | ||
|
||
:param s3_client: Client for s3 manipulation, default to domain S3Client | ||
:type s3_client: | ||
:param hathor_core_client: Client for make hathor-core requests, default to domain HathorCoreClient | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. HathorCoreClient? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes. It is the client to make requests for a full-node. |
||
:type hathor_core_client: | ||
""" | ||
def __init__( | ||
self, | ||
s3_client: Union[S3Client, None] = None, | ||
hathor_core_client: Union[HathorCoreClient, None] = None | ||
) -> None: | ||
self.s3_client = s3_client or S3Client() | ||
def __init__(self, hathor_core_client: Union[HathorCoreClient, None] = None) -> None: | ||
self.hathor_core_client = hathor_core_client or HathorCoreClient() | ||
|
||
def get_token_metadata_from_s3(self, file: str) -> Union[TokenMetadata, None]: | ||
"""Retrieve token metadata from a json file stored in s3 | ||
|
||
:param file: file name | ||
:type file: str | ||
:raises Exception: The name of the bucket used to store the jsons must be on config | ||
:return: TokenMetadata object | ||
:rtype: TokenMetadata | ||
""" | ||
token_metadata_bucket = TOKEN_METADATA_BUCKET | ||
|
||
if token_metadata_bucket is None: | ||
raise Exception('No bucket name in config') | ||
|
||
token_raw_metadata = self.s3_client.load_file(token_metadata_bucket, file) | ||
if token_raw_metadata is None: | ||
return None | ||
|
||
token_metadata = json.loads(token_raw_metadata) | ||
return TokenMetadata.from_dict(token_metadata) | ||
|
||
def get_token(self, id: str) -> Union[Token, None]: | ||
response = self.hathor_core_client.get(TOKEN_ENDPOINT, {'id': id}) | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You should add this in the TODO as well.
Create new buckets, migrate token info from old bucket to the new one and delete old bucket when everything is working fine.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's still missing some
data:image/s3,"s3://crabby-images/febec/febec73eb569b5d4822eeb06d4a77ecbb8a2f88b" alt="image"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it covers all, but i'll be more specific