abstract-solana 0.0.0.17__tar.gz → 0.0.0.19__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of abstract-solana might be problematic. Click here for more details.

Files changed (22) hide show
  1. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/PKG-INFO +1 -1
  2. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/setup.py +1 -1
  3. abstract_solana-0.0.0.19/src/abstract_solana/log_message_functions.py +155 -0
  4. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/pubkey_utils.py +2 -1
  5. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana.egg-info/PKG-INFO +1 -1
  6. abstract_solana-0.0.0.19/test/test_abstract_solana.py +18 -0
  7. abstract_solana-0.0.0.17/src/abstract_solana/log_message_functions.py +0 -90
  8. abstract_solana-0.0.0.17/test/test_abstract_solana.py +0 -29
  9. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/README.md +0 -0
  10. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/setup.cfg +0 -0
  11. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/__init__.py +0 -0
  12. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/account_key_utils.py +0 -0
  13. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/constants.py +0 -0
  14. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/genesis_functions.py +0 -0
  15. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/index_utils.py +0 -0
  16. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/price_utils.py +0 -0
  17. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/signature_data_parse.py +0 -0
  18. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana/utils.py +0 -0
  19. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana.egg-info/SOURCES.txt +0 -0
  20. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana.egg-info/dependency_links.txt +0 -0
  21. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana.egg-info/requires.txt +0 -0
  22. {abstract_solana-0.0.0.17 → abstract_solana-0.0.0.19}/src/abstract_solana.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: abstract_solana
3
- Version: 0.0.0.17
3
+ Version: 0.0.0.19
4
4
  Home-page: https://github.com/AbstractEndeavors/abstract_solana
5
5
  Author: putkoff
6
6
  Author-email: partners@abstractendeavors.com
@@ -4,7 +4,7 @@ with open("README.md", "r", encoding="utf-8") as fh:
4
4
  long_description = fh.read()
5
5
  setuptools.setup(
6
6
  name='abstract_solana',
7
- version='0.0.0.17',
7
+ version='0.0.0.19',
8
8
  author='putkoff',
9
9
  author_email='partners@abstractendeavors.com',
10
10
  description="",
@@ -0,0 +1,155 @@
1
+ from abstract_utilities import make_list
2
+ import json,pprint
3
+ from .price_functions import *
4
+ from .signature_data_parse import get_log_messages_from_txn,get_instructions_from_txn,get_inner_instructions_from_txn
5
+ from .account_key_utils import get_all_account_keys
6
+ from .constants import TOKEN_PROGRAM_ID
7
+ def ifListGetSection(list_obj,i=0):
8
+ if list_obj and isinstance(list_obj,list) and len(list_obj)>i:
9
+ list_obj = list_obj[i]
10
+ return list_obj
11
+ def get_logs_from_index(txnData,index=None):
12
+ if index is not None:
13
+ allLogs = get_log_messages_from_txn(txnData)
14
+ endLog = get_end_log_index(txnData,index)
15
+ return allLogs[index:endLog]
16
+ def get_program_ids(txnData):
17
+ allLogs = get_log_messages_from_txn(txnData)
18
+ return [log.split(' ')[1] for log in allLogs if 'invoke' in log]
19
+ def get_program_id_from_log(logs):
20
+ for log in make_list(logs):
21
+ if 'invoke' in log.lower():
22
+ return log.split(' ')[1]
23
+ def get_stack_height_from_logs(logs):
24
+ for log in make_list(logs):
25
+ if 'invoke' in log.lower():
26
+ return int(log.split(' ')[-1][1:-1])
27
+ def get_end_log_index(txnData,index):
28
+ allLogs = get_log_messages_from_txn(txnData)
29
+ i=0
30
+ for i,log in enumerate(allLogs[index+1:]):
31
+ if 'invoke' in log.lower():
32
+ return index+1+i
33
+ return len(allLogs)
34
+ def get_stack_height_from_logs(logs):
35
+ for log in make_list(logs):
36
+ if 'invoke' in log.lower():
37
+ return int(log.split(' ')[-1][1:-1])
38
+ def get_program_id_from_log(logs):
39
+ for log in make_list(logs):
40
+ if 'invoke' in log.lower():
41
+ return log.split(' ')[1]
42
+
43
+ def get_all_logs(txnData):
44
+ logits = []
45
+ allLogs = get_log_messages_from_txn(txnData)
46
+ for i,log in enumerate(allLogs):
47
+ if 'invoke' in log.lower():
48
+ logits.append([])
49
+ logits[-1].append(log)
50
+ start = 0
51
+ for i,log in enumerate(logits):
52
+ length = len(log)
53
+ end = start+length
54
+ logits[i]={"programId":get_program_id_from_log(log[0]),
55
+ "start":start,
56
+ "end":end-1,
57
+ 'stackHeight':get_stack_height_from_logs(log[0]) or 1,
58
+ 'events':[event.split(':')[-1] or 'unknown' for event in get_log_events(log)],
59
+ 'vars':[event.split(':')[1:] or 'unknown' for event in get_log_events(log)],
60
+ 'logs':log}
61
+ start = end
62
+ return logits
63
+ def split_log_for_instruction(log):
64
+ return log.split('log:')[-1].split('Instruction:')[-1]
65
+ def clean_split_string(string,delim=' '):
66
+ return delim.join([spl for spl in string.split(' ') if spl])
67
+ def get_log_events(logs=None,index=None):
68
+ return [clean_split_string(split_log_for_instruction(log)) for log in logs if 'log:' in log.lower() or 'instruction:' in log.lower()]
69
+ def get_instruction_accounts(instruction,txnData):
70
+ accounts = get_all_account_keys(txnData)
71
+ return [accounts[index] for index in instruction.get('accounts',[]) if index < len(accounts)]
72
+ def get_instruction_info(instruction,txnData,instruction_index=0):
73
+
74
+ stack_height = instruction.get('stackHeight') or 1
75
+ accounts = instruction.get('accounts',[])
76
+ associatedAccounts = get_instruction_accounts(instruction,txnData)
77
+ instruction_info = {'instruction_index':instruction_index,'accounts':accounts,'associatedAccounts':associatedAccounts,'instructionStackHeight':stack_height}
78
+
79
+ instruction_info.update(get_for_program_ids_info(txnData)[instruction_index])
80
+ return instruction_info
81
+ def process_instructions(instructions,txnData,start_index=1):
82
+ catalog = []
83
+ for i, inst in enumerate(instructions):
84
+ instruction_index = start_index-1 + i
85
+ instruction_info = get_instruction_info(inst,txnData,instruction_index)
86
+ catalog.append(instruction_info)
87
+ return catalog
88
+ def get_instructions_catalog(txnData,printIt=False,saveIt=False):
89
+ instructions = get_instructions_from_txn(txnData)
90
+ outer_instructions_catalog = process_instructions(instructions,txnData)
91
+ inner_instructions = get_inner_instructions_from_txn(txnData)
92
+ if inner_instructions:
93
+ inner_instructions_catalog = process_instructions(inner_instructions,txnData,start_index=len(instructions))
94
+ complete_catalog = outer_instructions_catalog+inner_instructions_catalog
95
+ if printIt:
96
+ pprint.pprint(complete_catalog)
97
+ if saveIt:
98
+ with open(saveIt, 'w') as f:
99
+ json.dump(complete_catalog, f, indent=4)
100
+ return complete_catalog
101
+
102
+ def find_in_catalog(string,txnData,programId=None):
103
+ complete_catalog = get_instructions_catalog(txnData)
104
+ return [txn for txn in complete_catalog if [event for event in txn['events'] if string.lower() in event.lower()]]
105
+ def findKeyValueIdInCatalog(key,value,txnData,programId=None):
106
+ complete_catalog = get_instructions_catalog(txnData)
107
+ if programId:
108
+ complete_catalog = findKeyValueIdInCatalog('programId',programId,txnData)
109
+ return [txn for txn in make_list(complete_catalog) if txn.get(key) == value]
110
+ def find_account_in_catalog(account,catalog):
111
+ return ifListGetSection([txn for txn in make_list(catalog) if account in txn.get('associatedAccounts')])
112
+ def associate_logs_with_instructions(txnData):
113
+ accountKeys = get_all_account_keys(txnData)
114
+ instructions = txnData['transaction']['message']['instructions']
115
+ innerInstructions = txnData['meta']['innerInstructions'][0]['instructions']
116
+ allLogs = txnData['meta']['logMessages']
117
+ for logIndex,log in enumerate(allLogs):
118
+ log_programId = log['programId']
119
+ log_stackHeight = log.get('stackHeight') # Default to 0 if stackHeight is missing
120
+ # Search for matching instructions by programId and stackHeight
121
+ for instIndex,allInstruction in enumerate([instructions,innerInstructions]):
122
+ for i,instruction in enumerate(allInstruction):
123
+ program_id_index = instruction.get('programIdIndex')
124
+ if program_id_index is not None:
125
+ instruction_program_id = accountKeys[program_id_index]
126
+ instruction_stack_height = instruction.get('stackHeight', 1)
127
+ if instruction_program_id == log_programId and instruction_stack_height == log_stackHeight:
128
+ # Add log data to the matching instruction
129
+ instruction['logs'] = log['logs']
130
+ instruction['event'] = None if log['events'] == [] else log['events'][0]
131
+ instruction['start'] = log['start']
132
+ instruction['end'] = log['end']
133
+ instruction['stackHeight'] = instruction_stack_height
134
+ instruction['programId'] = instruction_program_id
135
+ instruction['associatedAccounts'] = [accountKeys[index] for index in instruction['accounts']]
136
+ if instIndex == 0:
137
+ instructions[i] = instruction
138
+ else:
139
+ innerInstructions[i] = instruction
140
+ allLogs[logIndex].update(instruction)
141
+ txnData['transaction']['message']['instructions'] = instructions
142
+ txnData['meta']['innerInstructions'][0]['instructions'] = innerInstructions
143
+ txnData['meta']['logMessages'] = allLogs
144
+ return txnData
145
+ def update_initial_txnData(txnData):
146
+ accountKeys = get_all_account_keys(txnData)
147
+ txnData = update_balance_data(txnData)
148
+ txnData['transaction']['message']['instructions'] = [{**inst,"instructionIndex":instIndex,"programId":accountKeys[inst.get('programIdIndex')],"stackHeight":inst.get('stackHeight', 1),"associatedAccounts":[accountKeys[index] for index in inst['accounts']]} for instIndex,inst in enumerate(txnData['transaction']['message']['instructions'])]
149
+ txnData['meta']['innerInstructions'][0]['instructions'] = [{**inst,"instructionIndex":instIndex+len(txnData['transaction']['message']['instructions']),"programId":accountKeys[inst.get('programIdIndex')],"stackHeight":inst.get('stackHeight', 1),"associatedAccounts":[accountKeys[index] for index in inst['accounts']]} for instIndex,inst in enumerate(txnData['meta']['innerInstructions'][0]['instructions'])]
150
+ txnData['meta']['logMessages'] = get_all_logs(txnData)
151
+ return txnData
152
+ def get_for_program_ids_info(txnData):
153
+ txnData = update_initial_txnData(txnData)
154
+ txnData = associate_logs_with_instructions(txnData)
155
+ return txnData
@@ -1,6 +1,7 @@
1
1
  from solders.pubkey import Pubkey
2
2
  from solders.signature import Signature
3
3
  from spl.token.instructions import get_associated_token_address
4
+ TOKEN_PROGRAM_ID = "6EF8rrecthR5Dkzon8Nwu78hRvfCKubJ14M5uBEwF6P"
4
5
 
5
6
  def pubkey_find_program_address(string,address,programId):
6
7
  return Pubkey.find_program_address([str(string).encode(), bytes(get_pubkey(address))],get_pubkey(programId))
@@ -67,5 +68,5 @@ def derive_associated_bonding_curve(mint,programId=None):
67
68
  return get_associated_token_address(derive_bonding_curve(mint,programId)[0], get_pubkey(mint))
68
69
 
69
70
  def derive_bonding_curve(mint,programId=None):
70
- programId = programId or "TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA"
71
+ programId = programId or TOKEN_PROGRAM_ID
71
72
  return pubkey_find_program_address("bonding-curve",mint,programId)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: abstract_solana
3
- Version: 0.0.0.17
3
+ Version: 0.0.0.19
4
4
  Home-page: https://github.com/AbstractEndeavors/abstract_solana
5
5
  Author: putkoff
6
6
  Author-email: partners@abstractendeavors.com
@@ -0,0 +1,18 @@
1
+ from abstract_solana import *
2
+ from spl.token.instructions import create_associated_token_account, get_associated_token_address
3
+ from price_functions import *
4
+ from abstract_solcatcher import *
5
+ def get_solcatcher_api(endpoint,*args,**kwargs):
6
+ return get_async_response(get_solcatcher_endpoint,endpoint=endpoint,*args,**kwargs)
7
+
8
+ mint = "911eA3wRZ85ZiSTpmCH1hksPGLGgzyVpfMXtJ4zSzVJ5"
9
+ bonding_curve = str(derive_bonding_curve(mint)[0])
10
+ bonding_curve_signature = get_solcatcher_api("getGenesisSignature",account=str(bonding_curve))
11
+ transfers = []
12
+ txnData = get_solcatcher_api("getTransaction",tx_sig=str(bonding_curve_signature))
13
+ txnData = get_transfer_instructions(txnData)
14
+ input(txnData)
15
+ txnData = get_for_program_ids_info(txnData)
16
+ input(txnData)
17
+
18
+
@@ -1,90 +0,0 @@
1
- from abstract_utilities import make_list
2
- import json,pprint
3
- from .signature_data_parse import get_log_messages_from_txn,get_instructions_from_txn,get_inner_instructions_from_txn
4
- from .account_key_utils import get_all_account_keys
5
- from .constants import TOKEN_PROGRAM_ID
6
- def ifListGetSection(list_obj,i=0):
7
- if list_obj and isinstance(list_obj,list) and len(list_obj)>i:
8
- list_obj = list_obj[i]
9
- return list_obj
10
- def get_logs_from_index(txnData,index=None):
11
- if index is not None:
12
- allLogs = get_log_messages_from_txn(txnData)
13
- endLog = get_end_log_index(txnData,index)
14
- return allLogs[index:endLog]
15
- def get_program_ids(txnData):
16
- allLogs = get_log_messages_from_txn(txnData)
17
- return [log.split(' ')[1] for log in allLogs if 'invoke' in log]
18
- def get_program_id_from_log(logs):
19
- for log in make_list(logs):
20
- if 'invoke' in log.lower():
21
- return log.split(' ')[1]
22
- def get_stack_height_from_logs(logs):
23
- for log in make_list(logs):
24
- if 'invoke' in log.lower():
25
- return int(log.split(' ')[-1][1:-1])
26
- def get_end_log_index(txnData,index):
27
- allLogs = get_log_messages_from_txn(txnData)
28
- i=0
29
- for i,log in enumerate(allLogs[index+1:]):
30
- if 'invoke' in log.lower():
31
- return index+1+i
32
- return len(allLogs)
33
- def get_for_program_ids_info(txnData):
34
- allLogs = get_log_messages_from_txn(txnData)
35
- return [{"programId":get_program_id_from_log(allLogs[i]),
36
- "start":i,
37
- "end":get_end_log_index(txnData,i),
38
- 'stackHeight':get_stack_height_from_logs(allLogs[i]),
39
- 'events':[event.split(':')[0] for event in get_log_events(get_logs_from_index(txnData,index=i))],
40
- 'logs':get_logs_from_index(txnData,index=i)
41
- } for i in range(len(allLogs)) if 'invoke' in allLogs[i].lower()]
42
- def split_log_for_instruction(log):
43
- return log.split('log:')[-1].split('Instruction:')[-1]
44
- def clean_split_string(string,delim=' '):
45
- return delim.join([spl for spl in string.split(' ') if spl])
46
- def get_log_events(logs=None,index=None):
47
- return [clean_split_string(split_log_for_instruction(log)) for log in logs if 'log:' in log.lower() or 'instruction:' in log.lower()]
48
- def get_instruction_accounts(instruction,txnData):
49
- accounts = get_all_account_keys(txnData)
50
- return [accounts[index] for index in instruction.get('accounts',[]) if index < len(accounts)]
51
- def get_instruction_info(instruction,txnData,instruction_index=0):
52
-
53
- stack_height = instruction.get('stackHeight') or 1
54
- accounts = instruction.get('accounts',[])
55
- associatedAccounts = get_instruction_accounts(instruction,txnData)
56
- instruction_info = {'instruction_index':instruction_index,'accounts':accounts,'associatedAccounts':associatedAccounts,'instructionStackHeight':stack_height}
57
-
58
- instruction_info.update(get_for_program_ids_info(txnData)[instruction_index])
59
- return instruction_info
60
- def process_instructions(instructions,txnData,start_index=1):
61
- catalog = []
62
- for i, inst in enumerate(instructions):
63
- instruction_index = start_index-1 + i
64
- instruction_info = get_instruction_info(inst,txnData,instruction_index)
65
- catalog.append(instruction_info)
66
- return catalog
67
- def get_instructions_catalog(txnData,printIt=False,saveIt=False):
68
- instructions = get_instructions_from_txn(txnData)
69
- outer_instructions_catalog = process_instructions(instructions,txnData)
70
- inner_instructions = get_inner_instructions_from_txn(txnData)
71
- if inner_instructions:
72
- inner_instructions_catalog = process_instructions(inner_instructions,txnData,start_index=len(instructions))
73
- complete_catalog = outer_instructions_catalog+inner_instructions_catalog
74
- if printIt:
75
- pprint.pprint(complete_catalog)
76
- if saveIt:
77
- with open(saveIt, 'w') as f:
78
- json.dump(complete_catalog, f, indent=4)
79
- return complete_catalog
80
-
81
- def find_in_catalog(string,txnData,programId=None):
82
- complete_catalog = get_instructions_catalog(txnData)
83
- return [txn for txn in complete_catalog if [event for event in txn['events'] if string.lower() in event.lower()]]
84
- def findKeyValueIdInCatalog(key,value,txnData,programId=None):
85
- complete_catalog = get_instructions_catalog(txnData)
86
- if programId:
87
- complete_catalog = findKeyValueIdInCatalog('programId',programId,txnData)
88
- return [txn for txn in make_list(complete_catalog) if txn.get(key) == value]
89
- def find_account_in_catalog(account,catalog):
90
- return ifListGetSection([txn for txn in make_list(catalog) if account in txn.get('associatedAccounts')])
@@ -1,29 +0,0 @@
1
- from abstract_solana import *
2
- from spl.token.instructions import create_associated_token_account, get_associated_token_address
3
- from price_functions import *
4
- from abstract_solcatcher import *
5
- def get_solcatcher_api(endpoint,*args,**kwargs):
6
- return get_async_response(get_solcatcher_endpoint,endpoint=endpoint,*args,**kwargs)
7
- def get_balance_info(balanceInfo, programId, authority):
8
- return next((bal for bal in balanceInfo if programId == bal.get("programId") and authority == bal.get("owner")), {})
9
-
10
- def get_transfer_info(transfer_instructions, txnData):
11
- for transfer_instruction in transfer_instructions:
12
- updated_instruction = get_balance_from_instruction(transfer_instruction, txnData)
13
- transfer_instruction.update(updated_instruction)
14
- return transfer_instructions
15
-
16
- def get_transfer_instructions(txnData,programId=None):
17
- programId = programId or PUMP_FUN_ASSOC_TOKEN_ACC_PROG
18
- return get_transfer_info(find_in_catalog('transfer',txnData,programId=programId),txnData)
19
-
20
- mint = "911eA3wRZ85ZiSTpmCH1hksPGLGgzyVpfMXtJ4zSzVJ5"
21
- bondingCurve = str(derive_bonding_curve(mint))
22
- #input(getsignaturesforaddress(str(derive_bonding_curve(mint)[0])))
23
- signature = get_solcatcher_api("getGenesisSignature",account=str(derive_associated_bonding_curve(mint,PUMP_FUN_PROGRAM)[0]))
24
- txnData = get_solcatcher_api("getTransaction",tx_sig=str(signature))
25
- input(get_all_account_keys(txnData))
26
- parsed_txnData=parse_instruction_and_token_balances(txnData)
27
- input(parsed_txnData)
28
- input(find_in_catalog('Transfer',parsed_txnData,programId=PUMP_FUN_PROGRAM))
29
- input(get_transfer_instructions(txnData))