-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSuc_PS_DIS_AD_VS.rs
266 lines (225 loc) · 9.78 KB
/
Suc_PS_DIS_AD_VS.rs
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
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
use anchor_lang::prelude::*;
use anchor_lang::system_program;
#[allow(unused_imports)]
use pyth_sdk_solana::load_price_feed_from_account_info;
use anchor_spl::token::{self, Token, TokenAccount, Transfer};
use solana_program::{
account_info::AccountInfo,
pubkey::Pubkey,
clock::Clock,
};
declare_id!("4UjdrPr1Tv1974XZgLRZ63Wu4XisLRS2rh9K4ChK1wB7");
#[program]
pub mod presale_vesting {
use super::*;
pub fn initialize_presale(
ctx: Context<InitializePresale>,
token_mint: Pubkey,
admin: Pubkey,
bump: u8,
public_sale_price: u64 // Add public sale price
) -> Result<()> {
let presale = &mut ctx.accounts.presale_account;
presale.token_mint = token_mint;
presale.admin = admin;
presale.total_tokens_allocated = 0;
presale.is_closed = false;
presale.bump = bump; // Save the bump seed
presale.public_sale_price = public_sale_price; // Set the public sale price
Ok(())
}
pub fn contribute(ctx: Context<Contribute>, lamports_paid: u64) -> Result<()> {
let presale = &mut ctx.accounts.presale_account;
// Ensure the presale has not ended
require!(!presale.is_closed, CustomError::PresaleClosed);
// Calculate the discounted price (85% of public sale price)
let discounted_price = presale.public_sale_price * 85 / 100; // 15% discount
// Calculate the number of tokens the contributor receives
let tokens_to_allocate = lamports_paid / discounted_price; // Tokens = lamports paid / discounted price
require!(tokens_to_allocate > 0, CustomError::InvalidContribution);
// Transfer lamports from contributor to admin wallet (SOL payment)
**ctx.accounts.admin_wallet.to_account_info().try_borrow_mut_lamports()? += lamports_paid;
**ctx.accounts.contributor.to_account_info().try_borrow_mut_lamports()? -= lamports_paid;
// Update allocation account with the contributed amount
let allocation = &mut ctx.accounts.allocation_account;
allocation.amount += tokens_to_allocate; // Allocate tokens
allocation.cliff_timestamp = presale.cliff_timestamp;
allocation.vesting_end_timestamp = presale.vesting_end_timestamp;
presale.total_tokens_allocated += tokens_to_allocate;
Ok(())
}
pub fn claim_tokens(ctx: Context<ClaimTokens>, claimable_now: u64) -> Result<()> {
let allocation = &mut ctx.accounts.allocation_account;
// Create a longer-lived variable for the presale_account key
let presale_account_key = ctx.accounts.presale_account.key();
// Prepare seeds for signing
let seeds = &[
b"presale",
presale_account_key.as_ref(),
&[ctx.accounts.presale_account.bump],
];
let signer = &[&seeds[..]];
// Transfer tokens to the contributor
let cpi_accounts = Transfer {
from: ctx.accounts.presale_wallet.to_account_info(), // Token source
to: ctx.accounts.contributor_wallet.to_account_info(), // Token destination
authority: ctx.accounts.presale_account.to_account_info(), // Authority
};
let cpi_context = CpiContext::new_with_signer(
ctx.accounts.token_program.to_account_info(),
cpi_accounts,
signer,
);
token::transfer(cpi_context, claimable_now)?;
// Update allocation
allocation.claimed_amount += claimable_now;
Ok(())
}
/// New Airdrop Function for Automated Token Vesting
pub fn airdrop_tokens(ctx: Context<AirdropTokens>) -> Result<()> {
let allocation = &mut ctx.accounts.allocation_account;
let presale = &ctx.accounts.presale_account;
// Ensure the presale has ended
let current_time = Clock::get()?.unix_timestamp as u64;
require!(current_time >= allocation.cliff_timestamp, CustomError::CliffNotReached);
// Calculate claimable tokens based on the vesting schedule
let total_allocation = allocation.amount;
let upfront_allocation = total_allocation / 10; // 10% upfront
let monthly_allocation = total_allocation * 9 / 100; // 9% monthly
let months_elapsed = ((current_time - allocation.cliff_timestamp) / (30 * 24 * 60 * 60)) as u64;
let mut claimable_amount = 0;
if months_elapsed == 0 {
// If we're at the end of the cliff, release 10% upfront
claimable_amount = upfront_allocation;
} else if months_elapsed > 0 {
// Calculate the total claimable amount based on elapsed months
let max_months = 10; // 10 months of vesting
let months_to_claim = months_elapsed.min(max_months);
claimable_amount = upfront_allocation + (monthly_allocation * months_to_claim);
}
// Ensure we do not over-distribute tokens
claimable_amount = claimable_amount.saturating_sub(allocation.claimed_amount);
require!(claimable_amount > 0, CustomError::NothingToClaim);
// Prepare seeds for signing
let presale_key = presale.key();
let seeds = &[
b"presale",
presale_key.as_ref(),
&[presale.bump],
];
let signer = &[&seeds[..]];
// Transfer claimable tokens to the contributor
let cpi_accounts = Transfer {
from: ctx.accounts.presale_wallet.to_account_info(), // Token source
to: ctx.accounts.contributor_wallet.to_account_info(), // Token destination
authority: ctx.accounts.presale_account.to_account_info(), // Authority
};
let cpi_context = CpiContext::new_with_signer(
ctx.accounts.token_program.to_account_info(),
cpi_accounts,
signer,
);
token::transfer(cpi_context, claimable_amount)?;
// Update allocation to reflect claimed tokens
allocation.claimed_amount += claimable_amount;
Ok(())
}
pub fn close_presale(ctx: Context<ClosePresale>) -> Result<()> {
let presale = &mut ctx.accounts.presale_account;
// Only the admin can close the presale
require!(ctx.accounts.admin.key() == presale.admin, CustomError::Unauthorized);
presale.is_closed = true;
Ok(())
}
}
#[derive(Accounts)]
pub struct InitializePresale<'info> {
#[account(init, payer = admin, space = 8 + 32 + 32 + 8 + 8 + 1)]
pub presale_account: Account<'info, PresaleAccount>,
#[account(mut)]
pub admin: Signer<'info>,
#[account(address = system_program::ID)]
pub system_program: Program<'info, System>,
}
#[derive(Accounts)]
pub struct Contribute<'info> {
#[account(mut)]
pub presale_account: Account<'info, PresaleAccount>, // Presale state
#[account(
init_if_needed,
payer = contributor,
space = 8 + 8 + 8 + 8 + 8
)]
pub allocation_account: Account<'info, AllocationAccount>, // Allocation state for the contributor
#[account(mut)]
pub contributor: Signer<'info>, // Contributor wallet
/// CHECK: Admin wallet account (could add stricter validation here)
pub admin_wallet: AccountInfo<'info>, // Admin wallet to receive funds
#[account(address = token::ID)]
pub token_program: Program<'info, Token>, // Token program
pub system_program: Program<'info, System>, // System program
}
#[derive(Accounts)]
pub struct ClaimTokens<'info> {
#[account(mut)]
pub presale_account: Account<'info, PresaleAccount>, // Presale state
#[account(mut)]
pub allocation_account: Account<'info, AllocationAccount>, // Allocation state for the contributor
#[account(mut)]
pub presale_wallet: Account<'info, TokenAccount>, // Presale token wallet
#[account(mut)]
pub contributor_wallet: Account<'info, TokenAccount>, // Contributor token wallet
#[account(address = token::ID)]
pub token_program: Program<'info, Token>, // Token program
}
#[derive(Accounts)]
pub struct AirdropTokens<'info> {
#[account(mut)]
pub presale_account: Account<'info, PresaleAccount>, // Presale state
#[account(mut)]
pub allocation_account: Account<'info, AllocationAccount>, // Contributor's allocation
#[account(mut)]
pub presale_wallet: Account<'info, TokenAccount>, // Presale token wallet
#[account(mut)]
pub contributor_wallet: Account<'info, TokenAccount>, // Contributor token wallet
#[account(address = token::ID)]
pub token_program: Program<'info, Token>, // Token program
}
#[derive(Accounts)]
pub struct ClosePresale<'info> {
#[account(mut)]
pub presale_account: Account<'info, PresaleAccount>,
#[account(signer)]
pub admin: AccountInfo<'info>,
}
#[account]
pub struct PresaleAccount {
pub token_mint: Pubkey, // Token mint address
pub admin: Pubkey, // Admin address
pub total_tokens_allocated: u64, // Total tokens allocated
pub cliff_timestamp: u64, // Cliff timestamp for vesting
pub vesting_end_timestamp: u64, // Vesting end timestamp
pub is_closed: bool, // Whether the presale is closed
pub bump: u8, // PDA bump seed
pub public_sale_price: u64, // Token price in public sale (e.g., 1 token = X lamports)
}
#[account]
pub struct AllocationAccount {
pub amount: u64,
pub claimed_amount: u64,
pub cliff_timestamp: u64,
pub vesting_end_timestamp: u64,
}
#[error_code]
pub enum CustomError {
#[msg("The presale has already been closed.")]
PresaleClosed,
#[msg("The cliff period has not been reached yet.")]
CliffNotReached,
#[msg("You have no tokens to claim at this time.")]
NothingToClaim,
#[msg("Unauthorized action.")]
Unauthorized,
#[msg("Invalid contribution. You must contribute enough to purchase at least one token.")]
InvalidContribution, // New error variant
}