mango-explorer/mango/accountinfoconverter.py

105 lines
4.3 KiB
Python

# # ⚠ Warning
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
# LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
# NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
# WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
# SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
#
# [🥭 Mango Markets](https://mango.markets/) support is available at:
# [Docs](https://docs.mango.markets/)
# [Discord](https://discord.gg/67jySBhxrg)
# [Twitter](https://twitter.com/mangomarkets)
# [Github](https://github.com/blockworks-foundation)
# [Email](mailto:hello@blockworks.foundation)
import typing
from decimal import Decimal
from solana.publickey import PublicKey
from .account import Account
from .accountinfo import AccountInfo
from .addressableaccount import AddressableAccount
from .cache import Cache
from .context import Context
from .group import Group
from .layouts import layouts
from .lotsizeconverter import NullLotSizeConverter
from .openorders import OpenOrders
from .orderbookside import PerpOrderBookSide
from .perpeventqueue import PerpEventQueue
from .perpmarketdetails import PerpMarketDetails
from .serumeventqueue import SerumEventQueue
from .token import Instrument, Token
from .tokenbank import NodeBank, RootBank, TokenBank
# # 🥭 build_account_info_converter function
#
# Given a `Context` and an account type, returns a function that can take an `AccountInfo` and
# return one of our objects.
#
def build_account_info_converter(
context: Context, account_type: str
) -> typing.Callable[[AccountInfo], AddressableAccount]:
account_type_upper = account_type.upper()
if account_type_upper == "GROUP":
return lambda account_info: Group.parse_with_context(context, account_info)
elif account_type_upper == "ACCOUNT":
def account_loader(account_info: AccountInfo) -> Account:
layout_account = layouts.MANGO_ACCOUNT.parse(account_info.data)
group_address = layout_account.group
group: Group = Group.load(context, group_address)
cache: Cache = group.fetch_cache(context)
return Account.parse(account_info, group, cache)
return account_loader
elif account_type_upper == "OPENORDERS":
return lambda account_info: OpenOrders.parse(
account_info, Decimal(6), Decimal(6)
)
elif account_type_upper == "PERPEVENTQUEUE":
return lambda account_info: PerpEventQueue.parse(
account_info, NullLotSizeConverter()
)
elif account_type_upper == "SERUMEVENTQUEUE":
return lambda account_info: SerumEventQueue.parse(account_info)
elif account_type_upper == "CACHE":
return lambda account_info: Cache.parse(account_info)
elif account_type_upper == "ROOTBANK":
return lambda account_info: RootBank.parse(account_info)
elif account_type_upper == "NODEBANK":
return lambda account_info: NodeBank.parse(account_info)
elif account_type_upper == "PERPMARKETDETAILS":
def perp_market_details_loader(account_info: AccountInfo) -> PerpMarketDetails:
layout_perp_market_details = layouts.PERP_MARKET.parse(account_info.data)
group_address = layout_perp_market_details.group
group: Group = Group.load(context, group_address)
return PerpMarketDetails.parse(account_info, group)
return perp_market_details_loader
elif account_type_upper == "PERPORDERBOOKSIDE":
class __FakePerpMarketDetails(PerpMarketDetails):
def __init__(self) -> None:
self.base_instrument = Instrument(
"UNKNOWNBASE", "Unknown Base", Decimal(0)
)
self.quote_token = TokenBank(
Token("UNKNOWNQUOTE", "Unknown Quote", Decimal(0), PublicKey(0)),
PublicKey(0),
)
self.base_lot_size = Decimal(1)
self.quote_lot_size = Decimal(1)
return lambda account_info: PerpOrderBookSide.parse(
account_info, __FakePerpMarketDetails()
)
raise Exception(f"Could not find AccountInfo converter for type {account_type}.")