Взаимодействие с программой solana напрямую
Я пытаюсь использовать solana-py для взаимодействия с программой solana, о которой у меня нет документации.
Программа Solana Magic Eden NFT Marketplace . Я уже пытался получить информацию об этом от привязки , но у него нет доступных данных.
Моя цель: я хотел бы вывести NFT напрямую без веб-интерфейса.
Мой тест: в качестве примера я хотел бы перечислить этот NFT, которым я владею: Robber#04977.
Поскольку я ничего не знаю об этой программе, так как у них нет доступной документации, я изучил другую транзакцию и нашел ту, которую я пытался воссоздать, но с моим nft: Successful transaction.
Я изменил учетные записи, использовал свой закрытый ключ и создал транзакцию с помощью solana-py: my failed transaction.
См. редактирование ниже для текущего состояния.
Оригинальный код:
from solana.transaction import AccountMeta, Transaction, TransactionInstruction
from solana.rpc.types import TxOpts
from solana.account import Account
from solana.rpc.api import Client
from solana.publickey import PublicKey
from solana.keypair import Keypair
from getpass import getpass
import base58
# setup client
url = 'https://api.mainnet-beta.solana.com'
client = Client(url)
program = 'MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8'
# get account from private key
pwd = getpass('Chrome -> Phantom -> Settings -> Export private Key')
byte_array = base58.b58decode(pwd)
keypair = list(map(lambda b: int(str(b)), byte_array))[:]
initializer_account = Keypair(keypair[0:32])
# create transaction and sign it
txn = Transaction(recent_blockhash=client.get_recent_blockhash()['result']['value']['blockhash'], fee_payer=initializer_account.public_key)
txn.add(
TransactionInstruction(
keys=[
AccountMeta(pubkey=PublicKey(initializer_account.public_key), is_signer=True, is_writable=True),
AccountMeta(pubkey=PublicKey('GG24iCpytsz2nxei81FHyEyduQAxCAJHWkDLitwr9MxQ'), is_signer=False, is_writable=True),
AccountMeta(pubkey=PublicKey('3gS9AqTJ9adw23tZ87Hn1ccyYJ5KZ5tcoNQfYhCFu2e3'), is_signer=False, is_writable=True),
AccountMeta(pubkey=PublicKey('TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA'), is_signer=False, is_writable=False),
AccountMeta(pubkey=PublicKey('11111111111111111111111111111111'), is_signer=False, is_writable=False),
],
program_id=PublicKey('MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8'),
data=bytes.fromhex('96d480ba740183710094357700000000ff') # sell für 2 Solana
)
)
txn.sign(initializer_account)
rpc_response = client.send_transaction(
txn,
initializer_account,
opts=TxOpts(skip_preflight=True, skip_confirmation=False)
)
Ответ содержит идентификатор транзакции для моей неудачной транзакции.
{'jsonrpc': '2.0',
'result': 'NytmsBK59kckV3nGBsw6Vi9XAw8jkpkQGgHKCMYNFPYXLy57caNN7icNpMepofNsdncJ2BVziFJ82e8PKpH1EnV',
'id': 3}
Лог программы от solscan выглядит так:
#1 Magic Eden NFT Marketplace instruction
> Program Magic Eden NFT Marketplace consumed 5829 of 200000 compute units
> Program returned error: Could not create program address with signer seeds: Provided seeds do not result in a valid address
Возможно, это какие-то неверные данные, которые я отправляю в программу. Я просто искал исторические транзакции, которые были успешными, и использовал их для своей транзакции.
редактировать : приближается, но еще не сделано. Сейчас я также создаю инструкции по созданию учетной записи и установке полномочий. Но это делается последовательно, а не как внутренняя инструкция.
from solana.transaction import AccountMeta, Transaction, TransactionInstruction
from solana.rpc.types import TxOpts
from solana.account import Account
from solana.rpc.api import Client
from solana.publickey import PublicKey
from solana.rpc.commitment import Recent, Root
from solana.keypair import Keypair
from solana.system_program import create_account, CreateAccountParams
from getpass import getpass
import base58
from spl.token.instructions import set_authority, SetAuthorityParams, AuthorityType
LAMPORTS_PER_SOL = 1000000000
url = 'https://api.mainnet-beta.solana.com'
client = Client(url)
pwd = getpass('Chrome -> Phantom -> Settings -> Export private Key')
# setup of accounts
program = 'MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8'
# get int based keypair of account
byte_array = base58.b58decode(pwd)
keypair = list(map(lambda b: int(str(b)), byte_array))[:]
initializer_account = Keypair(keypair[0:32])
token_account_robber = PublicKey('GG24iCpytsz2nxei81FHyEyduQAxCAJHWkDLitwr9MxQ')
# inner instruction: create account
from_account, new_account = initializer_account.public_key, Keypair().public_key
inner_instruction1 = create_account(
CreateAccountParams(
from_pubkey=from_account, new_account_pubkey=new_account,
lamports=int(0.00144768*LAMPORTS_PER_SOL), space=1, program_id=PublicKey('MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8'))
)
# make all accounts of this non signer and non writeable
inner_instruction1.keys[0].is_signer=False
inner_instruction1.keys[0].is_writable=False
inner_instruction1.keys[1].is_signer=False
inner_instruction1.keys[1].is_writable=False
# inner instruction: set authority
inner_instruction2 = set_authority(
SetAuthorityParams(
program_id=PublicKey('TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA'),
account=token_account_robber,
authority=AuthorityType.ACCOUNT_OWNER,
current_authority=initializer_account.public_key,
new_authority=PublicKey('GUfCR9mK6azb9vcpsxgXyj7XRPAKJd4KMHTTVvtncGgp')
)
)
# combine all instructions
txn = Transaction(recent_blockhash=client.get_recent_blockhash()['result']['value']['blockhash'], fee_payer=initializer_account.public_key)
txn.add(
TransactionInstruction(
keys=[
AccountMeta(pubkey=PublicKey(initializer_account.public_key), is_signer=True, is_writable=True),
AccountMeta(pubkey=PublicKey('GG24iCpytsz2nxei81FHyEyduQAxCAJHWkDLitwr9MxQ'), is_signer=False, is_writable=True),
AccountMeta(pubkey=new_account, is_signer=False, is_writable=True),
AccountMeta(pubkey=PublicKey('TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA'), is_signer=False, is_writable=False),
AccountMeta(pubkey=PublicKey('11111111111111111111111111111111'), is_signer=False, is_writable=False),
],
program_id=PublicKey('MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8'),
data=bytes.fromhex('96d480ba740183710094357700000000ff') # sell für 2 Solana
)
)
txn.add(inner_instruction1)
txn.add(inner_instruction2)
# sign and send
txn.sign(initializer_account)
rpc_response = client.send_transaction(
txn,
initializer_account,
opts=TxOpts(skip_preflight=True, skip_confirmation=False)
)
'rpc_response` содержит следующее:
{'jsonrpc': '2.0',
'result': 'HTjMcuUHDoE3BkhcpWnLA6xWScDFzS7zQxWzUtffjMUkrZxaRjwrVjr8ta2Hr2uKxSUDXMzkLGiWbodgZk5DoEX',
'id': 190}
Лог от solscan:
#1 Magic Eden NFT Marketplace instruction
> Program Magic Eden NFT Marketplace consumed 8890 of 200000 compute units
> Program returned error: Could not create program address with signer seeds: Provided seeds do not result in a valid address
#2 System Program instruction
#3 Token Program instruction
Текущая проблема : добавленные к ошибке в журнале выше две новые инструкции (создать учетную запись и установить полномочия) не являются внутренними инструкциями, а просто некоторыми инструкциями, следующими за первой инструкцией.
Я использовал anchor-py, чтобы получить idl программы, но он недоступен (
anchorpy.error.IdlNotFoundError: IDL not found for program: MEisE1HzehtrDpAAT8PnLHjpSSkRYakotTuJRPjTpo8
).
2 ответа
Отказ от ответственности: я никогда не использовал solana-py, но, глядя на счета успешной транзакции, я вижу разницу. Account2 в успешной транзакции не является токеном NFT. т.е. https://solscan.io/account/5KCdkEiCeQQ7XxZfFgQ2h4CaaMu72386LX7daxcSUyFo , а при неудачной транзакции вы передаете https://solscan.io/token/3gS9AqTJ9adw23tZ87Hn1ccyYJ5KZ5tcoNQfYhCFu2e3 , который является вашим фактическим NFT.
благодаря этому твиту я нашел, как решить эту проблему:
anchorpy.error.IdlNotFoundError: IDL не найден для программы
После развертывания вашей программы Anchor вы также должны опубликовать ее IDL, используя:
anchor idl init <programId> -f <target/idl/program.json>