forked from drift-labs/driftpy
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathstart_lp.py
155 lines (127 loc) · 4.49 KB
/
start_lp.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
import sys
sys.path.append("../src/")
from driftpy.constants.config import configs
from anchorpy import Provider
import json
from anchorpy import Wallet
from solana.rpc.async_api import AsyncClient
from driftpy.drift_client import DriftClient
from driftpy.accounts import *
from solana.keypair import Keypair
# todo: airdrop udsc + init account for any kp
# rn do it through UI
from driftpy.drift_user import DriftUser
from driftpy.constants.numeric_constants import AMM_RESERVE_PRECISION
from solana.rpc import commitment
import pprint
from driftpy.constants.numeric_constants import QUOTE_PRECISION
async def view_logs(sig: str, connection: AsyncClient):
connection._commitment = commitment.Confirmed
logs = ""
try:
await connection.confirm_transaction(sig, commitment.Confirmed)
logs = (await connection.get_transaction(sig))["result"]["meta"]["logMessages"]
finally:
connection._commitment = commitment.Processed
pprint.pprint(logs)
async def main(
keypath,
env,
url,
market_index,
liquidity_amount,
operation,
):
with open(keypath, "r") as f:
secret = json.load(f)
kp = Keypair.from_secret_key(bytes(secret))
print("using public key:", kp.public_key)
print("market:", market_index)
config = configs[env]
wallet = Wallet(kp)
connection = AsyncClient(url)
provider = Provider(connection, wallet)
dc = DriftClient.from_config(config, provider)
drift_user = User(dc)
total_collateral = await drift_user.get_total_collateral()
print("total collateral:", total_collateral / QUOTE_PRECISION)
if total_collateral == 0:
print("cannot lp with 0 collateral")
return
market = await get_perp_market_account(dc.program, market_index)
lp_amount = liquidity_amount * AMM_RESERVE_PRECISION
lp_amount -= lp_amount % market.amm.order_step_size
lp_amount = int(lp_amount)
print("standardized lp amount:", lp_amount / AMM_RESERVE_PRECISION)
if lp_amount < market.amm.order_step_size:
print("lp amount too small - exiting...")
print(f"{operation}ing {lp_amount} lp shares...")
sig = None
if operation == "add":
resp = input("confirm adding liquidity: Y?")
if resp != "Y":
print("confirmation failed exiting...")
return
sig = await dc.add_liquidity(lp_amount, market_index)
print(sig)
elif operation == "remove":
resp = input("confirm removing liquidity: Y?")
if resp != "Y":
print("confirmation failed exiting...")
return
sig = await dc.remove_liquidity(lp_amount, market_index)
print(sig)
elif operation == "view":
pass
elif operation == "settle":
resp = input("confirm settling revenue to if stake: Y?")
if resp != "Y":
print("confirmation failed exiting...")
return
sig = await dc.settle_lp(dc.authority, market_index)
print(sig)
else:
return
if sig:
print("confirming tx...")
await connection.confirm_transaction(sig)
position = dc.get_perp_position(market_index)
market = await get_perp_market_account(dc.program, market_index)
percent_provided = (position.lp_shares / market.amm.sqrt_k) * 100
print(f"lp shares: {position.lp_shares}")
print(f"providing {percent_provided}% of total market liquidity")
print("done! :)")
if __name__ == "__main__":
import argparse
import os
parser = argparse.ArgumentParser()
parser.add_argument(
"--keypath", type=str, required=False, default=os.environ.get("ANCHOR_WALLET")
)
parser.add_argument("--env", type=str, default="devnet")
parser.add_argument("--amount", type=float, required=False)
parser.add_argument("--market", type=int, required=True)
parser.add_argument(
"--operation", choices=["remove", "add", "view", "settle"], required=True
)
args = parser.parse_args()
if args.keypath is None:
raise NotImplementedError("need to provide keypath or set ANCHOR_WALLET")
match args.env:
case "devnet":
url = "https://api.devnet.solana.com"
case "mainnet":
url = "https://api.mainnet-beta.solana.com"
case _:
raise NotImplementedError("only devnet/mainnet env supported")
import asyncio
asyncio.run(
main(
args.keypath,
args.env,
url,
args.market,
args.amount,
args.operation,
)
)