diff --git a/auction-server/api-types/src/opportunity.rs b/auction-server/api-types/src/opportunity.rs index a7a843aa..1a4c453c 100644 --- a/auction-server/api-types/src/opportunity.rs +++ b/auction-server/api-types/src/opportunity.rs @@ -60,7 +60,7 @@ pub struct OpportunityBidResult { #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] #[serde(rename_all = "lowercase")] pub enum ProgramSvm { - Phantom, + SwapKamino, Limo, } @@ -206,18 +206,15 @@ pub enum OpportunityCreateProgramParamsV1Svm { #[serde_as(as = "DisplayFromStr")] order_address: Pubkey, }, - /// Phantom program specific parameters for the opportunity. - #[serde(rename = "phantom")] - #[schema(title = "phantom")] - Phantom { + /// Kamino swap program specific parameters for the opportunity. + #[serde(rename = "kamino_swap")] + #[schema(title = "kamino_swap")] + KaminoSwap { + // TODO*: we should make this more generic, a la `Swap` /// The user wallet address which requested the quote from the wallet. #[schema(example = "DUcTi3rDyS5QEmZ4BNRBejtArmDCWaPYGfN44vBJXKL5", value_type = String)] #[serde_as(as = "DisplayFromStr")] user_wallet_address: Pubkey, - - /// The maximum slippage percentage that the user is willing to accept. - #[schema(example = 0.5, value_type = f64)] - maximum_slippage_percentage: f64, }, } @@ -311,19 +308,15 @@ pub enum OpportunityParamsV1ProgramSvm { #[serde_as(as = "DisplayFromStr")] order_address: Pubkey, }, - /// Phantom program specific parameters for the opportunity. - #[serde(rename = "phantom")] - #[schema(title = "phantom")] - Phantom { + /// Swap program specific parameters for the opportunity. + #[serde(rename = "swap")] + #[schema(title = "swap")] + Swap { /// The user wallet address which requested the quote from the wallet. #[schema(example = "DUcTi3rDyS5QEmZ4BNRBejtArmDCWaPYGfN44vBJXKL5", value_type = String)] #[serde_as(as = "DisplayFromStr")] user_wallet_address: Pubkey, - /// The maximum slippage percentage that the user is willing to accept. - #[schema(example = 0.5, value_type = f64)] - maximum_slippage_percentage: f64, - /// The permission account to be permitted by the ER contract for the opportunity execution of the protocol. #[schema(example = "DUcTi3rDyS5QEmZ4BNRBejtArmDCWaPYGfN44vBJXKL5", value_type = String)] #[serde_as(as = "DisplayFromStr")] @@ -334,11 +327,21 @@ pub enum OpportunityParamsV1ProgramSvm { #[serde_as(as = "DisplayFromStr")] router_account: Pubkey, - /// The token searcher will send. - sell_token: TokenAmountSvm, + /// Details about the tokens to be swapped. Either the input token amount or the output token amount must be specified. + #[schema(inline)] + tokens: QuoteTokens, + }, +} - /// The token searcher will receive. - buy_token: TokenAmountSvm, +#[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug, ToResponse)] +pub enum QuoteTokens { + InputTokenSpecified { + input_token: TokenAmountSvm, + output_token: Pubkey, + }, + OutputTokenSpecified { + input_token: Pubkey, + output_token: TokenAmountSvm, }, } @@ -466,40 +469,50 @@ pub struct OpportunityBidEvm { pub signature: Signature, } -/// Parameters needed to create a new opportunity from the Phantom wallet. +/// Parameters needed to create a new opportunity from the swap request. /// Auction server will extract the output token price for the auction. #[serde_as] #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] -pub struct QuoteCreatePhantomV1Svm { +pub struct QuoteCreateV1SvmParams { /// The user wallet address which requested the quote from the wallet. #[schema(example = "DUcTi3rDyS5QEmZ4BNRBejtArmDCWaPYGfN44vBJXKL5", value_type = String)] #[serde_as(as = "DisplayFromStr")] - pub user_wallet_address: Pubkey, + pub user_wallet_address: Pubkey, /// The token mint address of the input token. #[schema(example = "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v", value_type = String)] #[serde_as(as = "DisplayFromStr")] - pub input_token_mint: Pubkey, + pub input_token_mint: Pubkey, /// The token mint address of the output token. #[schema(example = "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v", value_type = String)] #[serde_as(as = "DisplayFromStr")] - pub output_token_mint: Pubkey, - /// The input token amount that the user wants to swap. - #[schema(example = 100)] - pub input_token_amount: u64, - /// The maximum slippage percentage that the user is willing to accept. - #[schema(example = 0.5)] - pub maximum_slippage_percentage: f64, + pub output_token_mint: Pubkey, + /// The token amount that the user wants to swap out of/into. + #[schema(inline)] + pub specified_token_amount: SpecifiedTokenAmount, + /// The router account to send referral fees to. + #[schema(example = "DUcTi3rDyS5QEmZ4BNRBejtArmDCWaPYGfN44vBJXKL5", value_type = String)] + #[serde_as(as = "DisplayFromStr")] + pub router: Pubkey, /// The chain id for creating the quote. #[schema(example = "solana", value_type = String)] - pub chain_id: ChainId, + pub chain_id: ChainId, } #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] -#[serde(tag = "program")] -pub enum QuoteCreateV1Svm { - #[serde(rename = "phantom")] - #[schema(title = "phantom")] - Phantom(QuoteCreatePhantomV1Svm), +#[serde(tag = "side")] +pub enum SpecifiedTokenAmount { + #[serde(rename = "input")] + #[schema(title = "input")] + InputToken { + #[schema(example = 100)] + amount: u64, + }, + #[serde(rename = "output")] + #[schema(title = "output")] + OutputToken { + #[schema(example = 50)] + amount: u64, + }, } #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] @@ -507,7 +520,7 @@ pub enum QuoteCreateV1Svm { pub enum QuoteCreateSvm { #[serde(rename = "v1")] #[schema(title = "v1")] - V1(QuoteCreateV1Svm), + V1(QuoteCreateV1SvmParams), } #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] @@ -522,20 +535,17 @@ pub struct QuoteV1Svm { /// The signed transaction for the quote to be executed on chain which is valid until the expiration time. #[schema(example = "SGVsbG8sIFdvcmxkIQ==", value_type = String)] #[serde(with = "crate::serde::transaction_svm")] - pub transaction: VersionedTransaction, + pub transaction: VersionedTransaction, /// The expiration time of the quote (in seconds since the Unix epoch). #[schema(example = 1_700_000_000_000_000i64, value_type = i64)] - pub expiration_time: i64, + pub expiration_time: i64, /// The input token amount that the user wants to swap. - pub input_token: TokenAmountSvm, + pub input_token: TokenAmountSvm, /// The output token amount that the user will receive. - pub output_token: TokenAmountSvm, - /// The maximum slippage percentage that the user is willing to accept. - #[schema(example = 0.5)] - pub maximum_slippage_percentage: f64, + pub output_token: TokenAmountSvm, /// The chain id for the quote. #[schema(example = "solana", value_type = String)] - pub chain_id: ChainId, + pub chain_id: ChainId, } #[derive(Serialize, Deserialize, ToSchema, Clone, PartialEq, Debug)] @@ -558,7 +568,7 @@ impl OpportunityCreateSvm { match self { OpportunityCreateSvm::V1(params) => match ¶ms.program_params { OpportunityCreateProgramParamsV1Svm::Limo { .. } => ProgramSvm::Limo, - OpportunityCreateProgramParamsV1Svm::Phantom { .. } => ProgramSvm::Phantom, + OpportunityCreateProgramParamsV1Svm::KaminoSwap { .. } => ProgramSvm::SwapKamino, }, } } diff --git a/auction-server/build.rs b/auction-server/build.rs index 55d8d001..9519d8fc 100644 --- a/auction-server/build.rs +++ b/auction-server/build.rs @@ -37,8 +37,14 @@ fn build_evm_contracts() { } const SUBMIT_BID_INSTRUCTION_SVM: &str = "submit_bid"; -const PERMISSION_ACCOUNT_SVM: &str = "permission"; -const ROUTER_ACCOUNT_SVM: &str = "router"; +const SUBMIT_BID_PERMISSION_ACCOUNT_SVM: &str = "permission"; +const SUBMIT_BID_ROUTER_ACCOUNT_SVM: &str = "router"; + +const SWAP_INSTRUCTION_SVM: &str = "swap"; +const SWAP_ROUTER_ACCOUNT_SVM: &str = "router_fee_receiver_ta"; +const SWAP_USER_WALLET_ACCOUNT_SVM: &str = "trader"; +const SWAP_MINT_INPUT_ACCOUNT_SVM: &str = "mint_input"; +const SWAP_MINT_OUTPUT_ACCOUNT_SVM: &str = "mint_output"; const IDL_LOCATION: &str = "../contracts/svm/target/idl/express_relay.json"; fn extract_account_position(idl: Idl, instruction_name: &str, account_name: &str) -> usize { @@ -71,7 +77,7 @@ fn verify_and_extract_idl_data() { extract_account_position( express_relay_idl.clone(), SUBMIT_BID_INSTRUCTION_SVM, - PERMISSION_ACCOUNT_SVM, + SUBMIT_BID_PERMISSION_ACCOUNT_SVM, ) ); println!( @@ -79,7 +85,39 @@ fn verify_and_extract_idl_data() { extract_account_position( express_relay_idl.clone(), SUBMIT_BID_INSTRUCTION_SVM, - ROUTER_ACCOUNT_SVM, + SUBMIT_BID_ROUTER_ACCOUNT_SVM, + ) + ); + println!( + "cargo::rustc-env=SWAP_ROUTER_ACCOUNT_POSITION={}", + extract_account_position( + express_relay_idl.clone(), + SWAP_INSTRUCTION_SVM, + SWAP_ROUTER_ACCOUNT_SVM, + ) + ); + println!( + "cargo::rustc-env=SWAP_USER_WALLET_ACCOUNT_POSITION={}", + extract_account_position( + express_relay_idl.clone(), + SWAP_INSTRUCTION_SVM, + SWAP_USER_WALLET_ACCOUNT_SVM, + ) + ); + println!( + "cargo::rustc-env=SWAP_MINT_INPUT_ACCOUNT_POSITION={}", + extract_account_position( + express_relay_idl.clone(), + SWAP_INSTRUCTION_SVM, + SWAP_MINT_INPUT_ACCOUNT_SVM, + ) + ); + println!( + "cargo::rustc-env=SWAP_MINT_OUTPUT_ACCOUNT_POSITION={}", + extract_account_position( + express_relay_idl.clone(), + SWAP_INSTRUCTION_SVM, + SWAP_MINT_OUTPUT_ACCOUNT_SVM, ) ); } diff --git a/auction-server/src/api.rs b/auction-server/src/api.rs index ce2a1423..7c82eb45 100644 --- a/auction-server/src/api.rs +++ b/auction-server/src/api.rs @@ -340,6 +340,7 @@ pub async fn start_api(run_options: RunOptions, store: Arc) -> Result< api_types::bid::Bids, api_types::SvmChainUpdate, + api_types::opportunity::SpecifiedTokenAmount, api_types::opportunity::OpportunityBidEvm, api_types::opportunity::OpportunityBidResult, api_types::opportunity::OpportunityMode, @@ -360,10 +361,10 @@ pub async fn start_api(run_options: RunOptions, store: Arc) -> Result< api_types::opportunity::OpportunityParamsV1Evm, api_types::opportunity::QuoteCreate, api_types::opportunity::QuoteCreateSvm, - api_types::opportunity::QuoteCreateV1Svm, - api_types::opportunity::QuoteCreatePhantomV1Svm, + api_types::opportunity::QuoteCreateV1SvmParams, api_types::opportunity::Quote, api_types::opportunity::QuoteSvm, + api_types::opportunity::QuoteTokens, api_types::opportunity::QuoteV1Svm, api_types::opportunity::OpportunityDelete, api_types::opportunity::OpportunityDeleteSvm, diff --git a/auction-server/src/auction/entities/auction.rs b/auction-server/src/auction/entities/auction.rs index 51ebbc56..ccaa6786 100644 --- a/auction-server/src/auction/entities/auction.rs +++ b/auction-server/src/auction/entities/auction.rs @@ -34,6 +34,7 @@ pub struct Auction { pub bids: Vec>, } +#[derive(PartialEq)] pub enum SubmitType { ByServer, ByOther, diff --git a/auction-server/src/auction/service/mod.rs b/auction-server/src/auction/service/mod.rs index 37591954..3ee83c67 100644 --- a/auction-server/src/auction/service/mod.rs +++ b/auction-server/src/auction/service/mod.rs @@ -85,10 +85,14 @@ pub mod verification; pub mod workers; pub struct ExpressRelaySvm { - pub program_id: Pubkey, - pub relayer: Keypair, - pub permission_account_position: usize, - pub router_account_position: usize, + pub program_id: Pubkey, + pub relayer: Keypair, + pub permission_account_position_submit_bid: usize, + pub router_account_position_submit_bid: usize, + pub router_account_position_swap: usize, + pub user_wallet_account_position_swap: usize, + pub mint_input_account_position_swap: usize, + pub mint_output_account_position_swap: usize, } pub struct ConfigSvm { diff --git a/auction-server/src/auction/service/verification.rs b/auction-server/src/auction/service/verification.rs index 55a782ce..649cc552 100644 --- a/auction-server/src/auction/service/verification.rs +++ b/auction-server/src/auction/service/verification.rs @@ -1,9 +1,6 @@ use { super::{ - auction_manager::{ - AuctionManager, - TOTAL_BIDS_PER_AUCTION_EVM, - }, + auction_manager::TOTAL_BIDS_PER_AUCTION_EVM, ChainTrait, Service, }, @@ -13,6 +10,7 @@ use { entities::{ self, BidChainData, + SubmitType, }, service::get_live_bids::GetLiveBidsInput, }, @@ -26,18 +24,25 @@ use { entities::{ Evm, PermissionKey, - PermissionKeySvm, Svm, }, traced_client::TracedClient, }, opportunity::{ self as opportunity, - service::get_live_opportunities::GetLiveOpportunitiesInput, + entities::{ + QuoteTokens, + TokenAmountSvm, + }, + service::{ + get_live_opportunities::GetLiveOpportunitiesInput, + get_quote::get_quote_permission_key, + }, }, }, ::express_relay::{ self as express_relay_svm, + FeeToken, }, anchor_lang::{ AnchorDeserialize, @@ -271,11 +276,17 @@ struct BidDataSvm { router: Pubkey, permission_account: Pubkey, deadline: OffsetDateTime, + submit_type: SubmitType, } const BID_MINIMUM_LIFE_TIME_SVM_SERVER: Duration = Duration::from_secs(5); const BID_MINIMUM_LIFE_TIME_SVM_OTHER: Duration = Duration::from_secs(10); +pub enum BidPaymentInstruction { + SubmitBid, + Swap, +} + impl Service { //TODO: merge this logic with simulator logic async fn query_lookup_table(&self, table: &Pubkey, index: usize) -> Result { @@ -391,7 +402,7 @@ impl Service { pub fn extract_submit_bid_data( instruction: &CompiledInstruction, ) -> Result { - let discriminator = express_relay_svm::instruction::SubmitBid::discriminator(); + let discriminator = express_relay_svm::instruction::SubmitBid::DISCRIMINATOR; express_relay_svm::SubmitBidArgs::try_from_slice( &instruction.data.as_slice()[discriminator.len()..], ) @@ -400,12 +411,30 @@ impl Service { }) } - // Checks that the transaction includes exactly one submit_bid instruction to the Express Relay on-chain program - pub fn verify_submit_bid_instruction( + pub fn extract_swap_data( + instruction: &CompiledInstruction, + ) -> Result { + let discriminator = express_relay_svm::instruction::Swap::DISCRIMINATOR; + express_relay_svm::SwapArgs::try_from_slice( + &instruction.data.as_slice()[discriminator.len()..], + ) + .map_err(|e| { + RestError::BadParameters(format!("Invalid submit_bid instruction data: {}", e)) + }) + } + + pub fn extract_express_relay_bid_instruction( &self, transaction: VersionedTransaction, + instruction_type: BidPaymentInstruction, ) -> Result { - let submit_bid_instructions: Vec = transaction + let discriminator = match instruction_type { + BidPaymentInstruction::SubmitBid => { + express_relay_svm::instruction::SubmitBid::DISCRIMINATOR + } + BidPaymentInstruction::Swap => express_relay_svm::instruction::Swap::DISCRIMINATOR, + }; + let instructions = transaction .message .instructions() .iter() @@ -415,31 +444,26 @@ impl Service { return false; } - instruction - .data - .starts_with(&express_relay_svm::instruction::SubmitBid::discriminator()) + instruction.data.starts_with(&discriminator) }) .cloned() - .collect(); + .collect::>(); - match submit_bid_instructions.len() { - 1 => Ok(submit_bid_instructions[0].clone()), - _ => Err(RestError::BadParameters( - "Bid has to include exactly one submit_bid instruction to Express Relay program" - .to_string(), - )), + match instructions.len() { + 1 => Ok(instructions[0].clone()), + _ => Err(RestError::BadParameters("Bid must include exactly one instruction to Express Relay program that pays the bid".to_string())), } } async fn check_deadline( &self, - permission_key: &PermissionKeySvm, + submit_type: &SubmitType, deadline: OffsetDateTime, ) -> Result<(), RestError> { - let minimum_bid_life_time = match self.get_submission_state(permission_key).await { - entities::SubmitType::ByServer => Some(BID_MINIMUM_LIFE_TIME_SVM_SERVER), - entities::SubmitType::ByOther => Some(BID_MINIMUM_LIFE_TIME_SVM_OTHER), - entities::SubmitType::Invalid => None, + let minimum_bid_life_time = match submit_type { + SubmitType::ByServer => Some(BID_MINIMUM_LIFE_TIME_SVM_SERVER), + SubmitType::ByOther => Some(BID_MINIMUM_LIFE_TIME_SVM_OTHER), + SubmitType::Invalid => None, }; match minimum_bid_life_time { @@ -465,42 +489,160 @@ impl Service { &self, transaction: VersionedTransaction, ) -> Result { - let submit_bid_instruction = self.verify_submit_bid_instruction(transaction.clone())?; - let submit_bid_data = Self::extract_submit_bid_data(&submit_bid_instruction)?; - - let permission_account = self - .extract_account( - &transaction, - &submit_bid_instruction, - self.config - .chain_config - .express_relay - .permission_account_position, - ) - .await?; - let router = self - .extract_account( - &transaction, - &submit_bid_instruction, - self.config - .chain_config - .express_relay - .router_account_position, - ) - .await?; - Ok(BidDataSvm { - amount: submit_bid_data.bid_amount, - permission_account, - router, - deadline: OffsetDateTime::from_unix_timestamp(submit_bid_data.deadline).map_err( - |e| { - RestError::BadParameters(format!( - "Invalid deadline: {:?} {:?}", - submit_bid_data.deadline, e - )) - }, - )?, - }) + let submit_bid_instruction_result = self.extract_express_relay_bid_instruction( + transaction.clone(), + BidPaymentInstruction::SubmitBid, + ); + let swap_instruction_result = self.extract_express_relay_bid_instruction( + transaction.clone(), + BidPaymentInstruction::Swap, + ); + + match ( + submit_bid_instruction_result.clone(), + swap_instruction_result.clone(), + ) { + (Ok(submit_bid_instruction), Err(_)) => { + let submit_bid_data = Self::extract_submit_bid_data(&submit_bid_instruction)?; + + let permission_account = self + .extract_account( + &transaction, + &submit_bid_instruction, + self.config + .chain_config + .express_relay + .permission_account_position_submit_bid, + ) + .await?; + let router = self + .extract_account( + &transaction, + &submit_bid_instruction, + self.config + .chain_config + .express_relay + .router_account_position_submit_bid, + ) + .await?; + if router == self.config.chain_config.wallet_program_router_account { + return Err(RestError::BadParameters( + "Using swap router account is not allowed for submit_bid instruction" + .to_string(), + )); + } + Ok(BidDataSvm { + amount: submit_bid_data.bid_amount, + permission_account, + router, + deadline: OffsetDateTime::from_unix_timestamp(submit_bid_data.deadline) + .map_err(|e| { + RestError::BadParameters(format!( + "Invalid deadline: {:?} {:?}", + submit_bid_data.deadline, e + )) + })?, + submit_type: SubmitType::ByServer, + }) + } + (Err(_), Ok(swap_instruction)) => { + let swap_data = Self::extract_swap_data(&swap_instruction)?; + + let router = self + .extract_account( + &transaction, + &swap_instruction, + self.config + .chain_config + .express_relay + .router_account_position_swap, + ) + .await?; + // TODO*: should work to remove this + if router != self.config.chain_config.wallet_program_router_account { + return Err(RestError::BadParameters( + "Must use approved router for swap instruction".to_string(), + )); + } + + let user_wallet = self + .extract_account( + &transaction, + &swap_instruction, + self.config + .chain_config + .express_relay + .user_wallet_account_position_swap, + ) + .await?; + + let mint_input = self + .extract_account( + &transaction, + &swap_instruction, + self.config + .chain_config + .express_relay + .mint_input_account_position_swap, + ) + .await?; + let mint_output = self + .extract_account( + &transaction, + &swap_instruction, + self.config + .chain_config + .express_relay + .mint_output_account_position_swap, + ) + .await?; + + let (bid_amount, tokens) = match swap_data.fee_token { + FeeToken::Input => ( + swap_data.amount_input, + QuoteTokens::InputTokenSpecified { + input_token: TokenAmountSvm { + token: mint_input, + amount: swap_data.amount_input, + }, + output_token: mint_output, + }, + ), + FeeToken::Output => ( + swap_data.amount_output, + QuoteTokens::OutputTokenSpecified { + input_token: mint_input, + output_token: TokenAmountSvm { + token: mint_output, + amount: swap_data.amount_output, + }, + }, + ), + }; + + let permission_account = get_quote_permission_key(&tokens, &user_wallet); + + Ok(BidDataSvm { + amount: bid_amount, + permission_account, + router, + // TODO*: to fix once deadline param added to swap instruction--just set this way to make sure compiles + deadline: OffsetDateTime::now_utc(), + // deadline: OffsetDateTime::from_unix_timestamp(swap_data.deadline).map_err( + // |e| { + // RestError::BadParameters(format!( + // "Invalid deadline: {:?} {:?}", + // swap_data.deadline, e + // )) + // }, + // )?, + submit_type: SubmitType::ByOther, + }) + } + _ => Err(RestError::BadParameters( + "Either submit_bid or swap must be present, but not both".to_string(), + )), + } } fn all_signatures_exists( @@ -540,20 +682,21 @@ impl Service { &self, bid: &entities::BidCreate, chain_data: &entities::BidChainDataSvm, + submit_type: &SubmitType, ) -> Result<(), RestError> { let message_bytes = chain_data.transaction.message.serialize(); let signatures = chain_data.transaction.signatures.clone(); let accounts = chain_data.transaction.message.static_account_keys(); let permission_key = chain_data.get_permission_key(); - match self.get_submission_state(&permission_key).await { - entities::SubmitType::Invalid => { + match submit_type { + SubmitType::Invalid => { // TODO Look at the todo comment in get_quote.rs file in opportunity module Err(RestError::BadParameters(format!( "The permission key is not valid for auction anymore: {:?}", permission_key ))) } - entities::SubmitType::ByOther => { + SubmitType::ByOther => { let opportunities = self .opportunity_service .get_live_opportunities(GetLiveOpportunitiesInput { @@ -573,7 +716,7 @@ impl Service { &opportunity.get_missing_signers(), ) } - entities::SubmitType::ByServer => { + SubmitType::ByServer => { self.all_signatures_exists(&message_bytes, accounts, &signatures, &[]) } } @@ -708,10 +851,10 @@ impl Verification for Service { }; let permission_key = bid_chain_data.get_permission_key(); tracing::Span::current().record("permission_key", bid_data.permission_account.to_string()); - self.check_deadline(&permission_key, bid_data.deadline) + self.check_deadline(&bid_data.submit_type, bid_data.deadline) + .await?; + self.verify_signatures(&bid, &bid_chain_data, &bid_data.submit_type) .await?; - self.verify_signatures(&bid, &bid_chain_data).await?; - // TODO we should verify that the wallet bids also include another instruction to the swap program with the appropriate accounts and fields self.simulate_bid(&bid).await?; // Check if the bid is not duplicate diff --git a/auction-server/src/config.rs b/auction-server/src/config.rs index cde529b3..4c322538 100644 --- a/auction-server/src/config.rs +++ b/auction-server/src/config.rs @@ -164,7 +164,7 @@ pub struct ConfigSvm { /// Timeout for RPC requests in seconds. #[serde(default = "default_rpc_timeout_svm")] pub rpc_timeout: u64, - /// The router account for Phantom. + /// The router account for swap program. // TODO*: we should work to remove this and fully identify swaps by ix type instead of router account #[serde_as(as = "DisplayFromStr")] pub wallet_program_router_account: Pubkey, #[serde(default)] diff --git a/auction-server/src/opportunity/api.rs b/auction-server/src/opportunity/api.rs index 6ef3c3a8..4e74335e 100644 --- a/auction-server/src/opportunity/api.rs +++ b/auction-server/src/opportunity/api.rs @@ -58,7 +58,7 @@ fn get_program(auth: &Auth) -> Result { match profile.name.as_str() { "limo" => Ok(ProgramSvm::Limo), - "phantom" => Ok(ProgramSvm::Phantom), + "kamino market" => Ok(ProgramSvm::SwapKamino), _ => Err(RestError::Forbidden), } } @@ -211,7 +211,7 @@ pub async fn post_quote( State(store): State>, Json(params): Json, ) -> Result, RestError> { - if get_program(&auth)? != ProgramSvm::Phantom { + if get_program(&auth)? != ProgramSvm::SwapKamino { return Err(RestError::Forbidden); } diff --git a/auction-server/src/opportunity/entities/opportunity_svm.rs b/auction-server/src/opportunity/entities/opportunity_svm.rs index baf9ffdc..fd48a4ab 100644 --- a/auction-server/src/opportunity/entities/opportunity_svm.rs +++ b/auction-server/src/opportunity/entities/opportunity_svm.rs @@ -32,15 +32,14 @@ pub struct OpportunitySvmProgramLimo { } #[derive(Debug, Clone, PartialEq)] -pub struct OpportunitySvmProgramWallet { - pub user_wallet_address: Pubkey, - pub maximum_slippage_percentage: f64, +pub struct OpportunitySvmProgramSwap { + pub user_wallet_address: Pubkey, } #[derive(Debug, Clone, PartialEq)] pub enum OpportunitySvmProgram { Limo(OpportunitySvmProgramLimo), - Phantom(OpportunitySvmProgramWallet), + SwapKamino(OpportunitySvmProgramSwap), } #[derive(Debug, Clone, PartialEq)] @@ -93,11 +92,10 @@ impl Opportunity for OpportunitySvm { }, ) } - OpportunitySvmProgram::Phantom(program) => { - repository::OpportunityMetadataSvmProgram::Phantom( - repository::OpportunityMetadataSvmProgramWallet { - user_wallet_address: program.user_wallet_address, - maximum_slippage_percentage: program.maximum_slippage_percentage, + OpportunitySvmProgram::SwapKamino(program) => { + repository::OpportunityMetadataSvmProgram::SwapKamino( + repository::OpportunityMetadataSvmProgramSwap { + user_wallet_address: program.user_wallet_address, }, ) } @@ -166,33 +164,45 @@ impl From for api::Opportunity { impl From for api::OpportunitySvm { fn from(val: OpportunitySvm) -> Self { let program = match val.program.clone() { - OpportunitySvmProgram::Limo(prgoram) => api::OpportunityParamsV1ProgramSvm::Limo { - order: prgoram.order, - order_address: prgoram.order_address, + OpportunitySvmProgram::Limo(program) => api::OpportunityParamsV1ProgramSvm::Limo { + order: program.order, + order_address: program.order_address, }, - OpportunitySvmProgram::Phantom(program) => { - api::OpportunityParamsV1ProgramSvm::Phantom { - user_wallet_address: program.user_wallet_address, - maximum_slippage_percentage: program.maximum_slippage_percentage, - permission_account: val.permission_account, - router_account: val.router, + OpportunitySvmProgram::SwapKamino(program) => { + let buy_token = val + .buy_tokens + .first() + .ok_or(anyhow::anyhow!( + "Failed to get buy token from opportunity svm" + )) + .expect("Failed to get buy token from opportunity svm"); + let sell_token = val + .sell_tokens + .first() + .ok_or(anyhow::anyhow!( + "Failed to get sell token from opportunity svm" + )) + .expect("Failed to get sell token from opportunity svm"); + let tokens = if buy_token.amount == 0 { + api::QuoteTokens::OutputTokenSpecified { + input_token: buy_token.token, + output_token: sell_token.clone().into(), + } + } else { + if sell_token.amount != 0 { + tracing::error!(opportunity = ?val, "Both token amounts are specified for swap opportunity"); + } + api::QuoteTokens::InputTokenSpecified { + input_token: buy_token.clone().into(), + output_token: sell_token.token, + } + }; + api::OpportunityParamsV1ProgramSvm::Swap { + user_wallet_address: program.user_wallet_address, + permission_account: val.permission_account, + router_account: val.router, // TODO can we make it type safe? - sell_token: val - .sell_tokens - .first() - .map(|t| t.clone().into()) - .ok_or(anyhow::anyhow!( - "Failed to get sell token from opportunity svm" - )) - .expect("Failed to get sell token from opportunity svm"), - buy_token: val - .sell_tokens - .first() - .map(|t| t.clone().into()) - .ok_or(anyhow::anyhow!( - "Failed to get sell token from opportunity svm" - )) - .expect("Failed to get sell token from opportunity svm"), + tokens, } } }; @@ -237,10 +247,9 @@ impl TryFrom> for Op order_address: program.order_address, }) } - repository::OpportunityMetadataSvmProgram::Phantom(program) => { - OpportunitySvmProgram::Phantom(OpportunitySvmProgramWallet { - user_wallet_address: program.user_wallet_address, - maximum_slippage_percentage: program.maximum_slippage_percentage, + repository::OpportunityMetadataSvmProgram::SwapKamino(program) => { + OpportunitySvmProgram::SwapKamino(OpportunitySvmProgramSwap { + user_wallet_address: program.user_wallet_address, }) } }; @@ -273,12 +282,10 @@ impl From for OpportunityCreateSvm { order, order_address, }), - api::OpportunityCreateProgramParamsV1Svm::Phantom { + api::OpportunityCreateProgramParamsV1Svm::KaminoSwap { user_wallet_address, - maximum_slippage_percentage, - } => OpportunitySvmProgram::Phantom(OpportunitySvmProgramWallet { + } => OpportunitySvmProgram::SwapKamino(OpportunitySvmProgramSwap { user_wallet_address, - maximum_slippage_percentage, }), }; @@ -322,7 +329,7 @@ impl From for OpportunityCreateSvm { impl OpportunitySvm { pub fn get_missing_signers(&self) -> Vec { match self.program.clone() { - OpportunitySvmProgram::Phantom(data) => vec![data.user_wallet_address], + OpportunitySvmProgram::SwapKamino(data) => vec![data.user_wallet_address], OpportunitySvmProgram::Limo(_) => vec![], } } @@ -336,7 +343,7 @@ impl From for api::ProgramSvm { fn from(val: OpportunitySvmProgram) -> Self { match val { OpportunitySvmProgram::Limo(_) => api::ProgramSvm::Limo, - OpportunitySvmProgram::Phantom(_) => api::ProgramSvm::Phantom, + OpportunitySvmProgram::SwapKamino(_) => api::ProgramSvm::SwapKamino, } } } diff --git a/auction-server/src/opportunity/entities/quote.rs b/auction-server/src/opportunity/entities/quote.rs index 7d4ffe11..d11df471 100644 --- a/auction-server/src/opportunity/entities/quote.rs +++ b/auction-server/src/opportunity/entities/quote.rs @@ -10,38 +10,62 @@ use { #[derive(Debug, Clone, PartialEq)] pub struct Quote { - pub transaction: VersionedTransaction, + pub transaction: VersionedTransaction, // The expiration time of the quote (in seconds since the Unix epoch) - pub expiration_time: i64, - pub input_token: TokenAmountSvm, - pub output_token: TokenAmountSvm, - pub maximum_slippage_percentage: f64, - pub chain_id: ChainId, + pub expiration_time: i64, + pub input_token: TokenAmountSvm, + pub output_token: TokenAmountSvm, + pub chain_id: ChainId, } #[derive(Debug, Clone, PartialEq)] pub struct QuoteCreate { - pub user_wallet_address: Pubkey, - pub input_token: TokenAmountSvm, - pub output_mint_token: Pubkey, - pub maximum_slippage_percentage: f64, - pub chain_id: ChainId, + pub user_wallet_address: Pubkey, + pub tokens: QuoteTokens, + pub router: Pubkey, + pub chain_id: ChainId, +} + +#[derive(Debug, Clone, PartialEq)] +pub enum QuoteTokens { + InputTokenSpecified { + input_token: TokenAmountSvm, + output_token: Pubkey, + }, + OutputTokenSpecified { + input_token: Pubkey, + output_token: TokenAmountSvm, + }, } impl From for QuoteCreate { fn from(quote_create: api::QuoteCreate) -> Self { - let api::QuoteCreate::Svm(api::QuoteCreateSvm::V1(api::QuoteCreateV1Svm::Phantom(params))) = - quote_create; + let api::QuoteCreate::Svm(api::QuoteCreateSvm::V1(params)) = quote_create; - Self { - user_wallet_address: params.user_wallet_address, - input_token: TokenAmountSvm { - token: params.input_token_mint, - amount: params.input_token_amount, + let tokens = match params.specified_token_amount { + api::SpecifiedTokenAmount::InputToken { amount } => QuoteTokens::InputTokenSpecified { + input_token: TokenAmountSvm { + token: params.input_token_mint, + amount, + }, + output_token: params.output_token_mint, }, - output_mint_token: params.output_token_mint, - maximum_slippage_percentage: params.maximum_slippage_percentage, - chain_id: params.chain_id, + api::SpecifiedTokenAmount::OutputToken { amount } => { + QuoteTokens::OutputTokenSpecified { + input_token: params.input_token_mint, + output_token: TokenAmountSvm { + token: params.output_token_mint, + amount, + }, + } + } + }; + + Self { + user_wallet_address: params.user_wallet_address, + tokens, + router: params.router, + chain_id: params.chain_id, } } } @@ -49,12 +73,11 @@ impl From for QuoteCreate { impl From for api::Quote { fn from(quote: Quote) -> Self { api::Quote::Svm(api::QuoteSvm::V1(api::QuoteV1Svm { - transaction: quote.transaction, - expiration_time: quote.expiration_time, - input_token: quote.input_token.into(), - output_token: quote.output_token.into(), - maximum_slippage_percentage: quote.maximum_slippage_percentage, - chain_id: quote.chain_id, + transaction: quote.transaction, + expiration_time: quote.expiration_time, + input_token: quote.input_token.into(), + output_token: quote.output_token.into(), + chain_id: quote.chain_id, })) } } diff --git a/auction-server/src/opportunity/repository/models.rs b/auction-server/src/opportunity/repository/models.rs index aabcd116..3270d642 100644 --- a/auction-server/src/opportunity/repository/models.rs +++ b/auction-server/src/opportunity/repository/models.rs @@ -57,17 +57,16 @@ pub struct OpportunityMetadataSvmProgramLimo { #[serde_as] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] -pub struct OpportunityMetadataSvmProgramWallet { +pub struct OpportunityMetadataSvmProgramSwap { #[serde_as(as = "DisplayFromStr")] - pub user_wallet_address: Pubkey, - pub maximum_slippage_percentage: f64, + pub user_wallet_address: Pubkey, } #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[serde(tag = "program", rename_all = "lowercase")] pub enum OpportunityMetadataSvmProgram { Limo(OpportunityMetadataSvmProgramLimo), - Phantom(OpportunityMetadataSvmProgramWallet), + SwapKamino(OpportunityMetadataSvmProgramSwap), } #[serde_as] diff --git a/auction-server/src/opportunity/service/estimate_price.rs b/auction-server/src/opportunity/service/estimate_price.rs deleted file mode 100644 index 81f342f4..00000000 --- a/auction-server/src/opportunity/service/estimate_price.rs +++ /dev/null @@ -1,23 +0,0 @@ -use { - super::{ - ChainTypeSvm, - Service, - }, - crate::{ - api::RestError, - opportunity::entities, - }, -}; - -pub struct EstimatePriceInput { - #[allow(dead_code)] - pub quote_create: entities::QuoteCreate, -} - -impl Service { - #[tracing::instrument(skip_all)] - pub async fn estimate_price(&self, _input: EstimatePriceInput) -> Result { - // TODO implement - return Ok(0); - } -} diff --git a/auction-server/src/opportunity/service/get_quote.rs b/auction-server/src/opportunity/service/get_quote.rs index 6f7e2f79..0518ef34 100644 --- a/auction-server/src/opportunity/service/get_quote.rs +++ b/auction-server/src/opportunity/service/get_quote.rs @@ -8,6 +8,7 @@ use { auction::{ entities::{ Auction, + BidStatus, BidStatusAuction, }, service::{ @@ -16,24 +17,18 @@ use { get_live_bids::GetLiveBidsInput, update_bid_status::UpdateBidStatusInput, update_submitted_auction::UpdateSubmittedAuctionInput, + verification::BidPaymentInstruction, Service as AuctionService, }, }, - kernel::entities::{ - PermissionKeySvm, - Svm, - }, + kernel::entities::PermissionKeySvm, opportunity::{ entities, - service::{ - add_opportunity::AddOpportunityInput, - estimate_price::EstimatePriceInput, - }, + service::add_opportunity::AddOpportunityInput, }, }, axum_prometheus::metrics, futures::future::join_all, - rand::Rng, solana_sdk::{ clock::Slot, pubkey::Pubkey, @@ -50,15 +45,69 @@ pub struct GetQuoteInput { pub quote_create: entities::QuoteCreate, } +pub fn get_quote_permission_key( + tokens: &entities::QuoteTokens, + user_wallet_address: &Pubkey, +) -> Pubkey { + // get pda seeded by user_wallet_address, mints, and token amount + let input_token_amount: [u8; 8]; + let output_token_amount: [u8; 8]; + let seeds = match tokens { + entities::QuoteTokens::InputTokenSpecified { + input_token, + output_token, + } => { + let input_token_mint = input_token.token.as_ref(); + let output_token_mint = output_token.as_ref(); + input_token_amount = input_token.amount.to_le_bytes(); + [ + user_wallet_address.as_ref(), + input_token_mint, + input_token_amount.as_ref(), + output_token_mint, + ] + } + entities::QuoteTokens::OutputTokenSpecified { + input_token, + output_token, + } => { + let input_token_mint = input_token.as_ref(); + let output_token_mint = output_token.token.as_ref(); + output_token_amount = output_token.amount.to_le_bytes(); + [ + user_wallet_address.as_ref(), + input_token_mint, + output_token_mint, + output_token_amount.as_ref(), + ] + } + }; + // since this permission key will not be used on-chain, we don't need to use the express relay program_id. + // we can use a distinctive bytes object for the program_id + Pubkey::find_program_address(&seeds, &Pubkey::default()).0 +} + impl Service { async fn get_opportunity_create_for_quote( &self, quote_create: entities::QuoteCreate, - output_amount: u64, ) -> Result { - let chain_config = self.get_config("e_create.chain_id)?; - let router = chain_config.wallet_program_router_account; - let permission_account = Pubkey::new_from_array(rand::thread_rng().gen()); + let router = quote_create.router; + let permission_account = + get_quote_permission_key("e_create.tokens, "e_create.user_wallet_address); + + // TODO*: we should fix the Opportunity struct (or create a new format) to more clearly distinguish Swap opps from traditional opps + // currently, we are using the same struct and just setting the unspecified token amount to 0 + let (input_mint, input_amount, output_mint, output_amount) = match quote_create.tokens { + entities::QuoteTokens::InputTokenSpecified { + input_token, + output_token, + } => (input_token.token, input_token.amount, output_token, 0), + entities::QuoteTokens::OutputTokenSpecified { + input_token, + output_token, + } => (input_token, 0, output_token.token, output_token.amount), + }; let core_fields = entities::OpportunityCoreFieldsCreate { permission_key: entities::OpportunitySvm::get_permission_key( @@ -66,9 +115,12 @@ impl Service { permission_account, ), chain_id: quote_create.chain_id, - sell_tokens: vec![quote_create.input_token], + sell_tokens: vec![entities::TokenAmountSvm { + token: input_mint, + amount: input_amount, + }], buy_tokens: vec![entities::TokenAmountSvm { - token: quote_create.output_mint_token, + token: output_mint, amount: output_amount, }], }; @@ -77,13 +129,12 @@ impl Service { core_fields, router, permission_account, - program: entities::OpportunitySvmProgram::Phantom( - entities::OpportunitySvmProgramWallet { - user_wallet_address: quote_create.user_wallet_address, - maximum_slippage_percentage: quote_create.maximum_slippage_percentage, + program: entities::OpportunitySvmProgram::SwapKamino( + entities::OpportunitySvmProgramSwap { + user_wallet_address: quote_create.user_wallet_address, }, ), - // TODO extract latest slot + // TODO* extract latest slot slot: Slot::default(), }) } @@ -93,22 +144,24 @@ impl Service { let config = self.get_config(&input.quote_create.chain_id)?; let auction_service = config.get_auction_service().await; - // TODO Check for the input amount tracing::info!(quote_create = ?input.quote_create, "Received request to get quote"); - let output_amount = self - .estimate_price(EstimatePriceInput { - quote_create: input.quote_create.clone(), - }) - .await?; let opportunity_create = self - .get_opportunity_create_for_quote(input.quote_create.clone(), output_amount) + .get_opportunity_create_for_quote(input.quote_create.clone()) .await?; let opportunity = self .add_opportunity(AddOpportunityInput { opportunity: opportunity_create, }) .await?; + let input_token = opportunity.buy_tokens[0].clone(); + let output_token = opportunity.sell_tokens[0].clone(); + if input_token.amount == 0 && output_token.amount == 0 { + tracing::error!(opportunity = ?opportunity, "Both token amounts are zero for swap opportunity"); + return Err(RestError::BadParameters( + "Both token amounts are zero for swap opportunity".to_string(), + )); + } // NOTE: This part will be removed after refactoring the permission key type let slice: [u8; 64] = opportunity @@ -123,7 +176,7 @@ impl Service { let bid_collection_time = OffsetDateTime::now_utc(); let mut bids = auction_service .get_live_bids(GetLiveBidsInput { - permission_key: permission_key_svm, + permission_key: permission_key_svm.clone(), }) .await; @@ -135,7 +188,7 @@ impl Service { // Add metrics let labels = [ ("chain_id", input.quote_create.chain_id.to_string()), - ("wallet", "phantom".to_string()), + ("router", input.quote_create.router.to_string()), ("total_bids", total_bids), ]; metrics::counter!("get_quote_total_bids", &labels).increment(1); @@ -146,24 +199,35 @@ impl Service { return Err(RestError::QuoteNotFound); } - // Find winner bid: the bid with the highest bid amount - bids.sort_by(|a, b| b.amount.cmp(&a.amount)); + // Find winner bid: + match input.quote_create.tokens { + entities::QuoteTokens::InputTokenSpecified { .. } => { + // highest bid = best (most output token returned) + bids.sort_by(|a, b| b.amount.cmp(&a.amount)); + } + entities::QuoteTokens::OutputTokenSpecified { .. } => { + // lowest bid = best (least input token consumed) + bids.sort_by(|a, b| a.amount.cmp(&b.amount)); + } + } let winner_bid = bids.first().expect("failed to get first bid"); - // Find the submit bid instruction from bid transaction to extract the deadline - let submit_bid_instruction = auction_service - .verify_submit_bid_instruction(winner_bid.chain_data.transaction.clone()) + let swap_instruction = auction_service + .extract_express_relay_bid_instruction( + winner_bid.chain_data.transaction.clone(), + BidPaymentInstruction::Swap, + ) .map_err(|e| { - tracing::error!("Failed to verify submit bid instruction: {:?}", e); + tracing::error!("Failed to verify swap instruction: {:?}", e); RestError::TemporarilyUnavailable })?; - let submit_bid_data = AuctionService::::extract_submit_bid_data( - &submit_bid_instruction, - ) - .map_err(|e| { - tracing::error!("Failed to extract submit bid data: {:?}", e); + let _swap_data = AuctionService::extract_swap_data(&swap_instruction).map_err(|e| { + tracing::error!("Failed to extract swap data: {:?}", e); RestError::TemporarilyUnavailable })?; + let deadline = i64::MAX; + // TODO*: to fix once deadline param added to swap instruction--just set this way to make sure compiles + // let deadline = swap_data.deadline; // Bids is not empty let auction = Auction::try_new(bids.clone(), bid_collection_time) @@ -184,55 +248,63 @@ impl Service { }) .await?; - self.task_tracker.spawn({ - let (repo, db, winner_bid) = (self.repo.clone(), self.db.clone(), winner_bid.clone()); - let auction_service = auction_service.clone(); - async move { - join_all(auction.bids.iter().map(|bid| { - auction_service.update_bid_status(UpdateBidStatusInput { - new_status: AuctionService::get_new_status( - bid, - &vec![winner_bid.clone()], - BidStatusAuction { - tx_hash: signature, - id: auction.id, - }, - ), - bid: bid.clone(), - }) - })) - .await; - // Remove opportunity to prevent further bids - // The handle auction loop will take care of the bids that were submitted late - - // TODO - // Maybe we should add state for opportunity. - // Right now logic for removing halted/expired bids, checks if opportunity exists. - // We should remove opportunity only after the auction bid result is broadcasted. - // This is to make sure we are not gonna remove the bids that are currently in the auction in the handle_auction loop. - let removal_reason = - entities::OpportunityRemovalReason::Invalid(RestError::InvalidOpportunity( - "Auction finished for the opportunity".to_string(), - )); - if let Err(e) = repo - .remove_opportunity(&db, &opportunity, removal_reason) - .await - { - tracing::error!("Failed to remove opportunity: {:?}", e); - } - } - }); + join_all(auction.bids.iter().map(|bid| { + auction_service.update_bid_status(UpdateBidStatusInput { + new_status: AuctionService::get_new_status( + bid, + &vec![winner_bid.clone()], + BidStatusAuction { + tx_hash: signature, + id: auction.id, + }, + ), + bid: bid.clone(), + }) + })) + .await; + // Remove opportunity to prevent further bids + // The handle auction loop will take care of the bids that were submitted late + + // TODO + // Maybe we should add state for opportunity. + // Right now logic for removing halted/expired bids, checks if opportunity exists. + // We should remove opportunity only after the auction bid result is broadcasted. + // This is to make sure we are not gonna remove the bids that are currently in the auction in the handle_auction loop. + let removal_reason = entities::OpportunityRemovalReason::Invalid( + RestError::InvalidOpportunity("Auction finished for the opportunity".to_string()), + ); + if let Err(e) = self + .repo + .remove_opportunity(&self.db, &opportunity, removal_reason) + .await + { + tracing::error!("Failed to remove opportunity: {:?}", e); + } + + // we check the winner bid status here to make sure the winner bid was successfully entered into the db as submitted + // this is because: if the winner bid was not successfully entered as submitted, that could indicate the presence of + // duplicate auctions for the same quote. in such a scenario, one auction will conclude first and update the bid status + // of its winner bid, and we want to ensure that a winner bid whose status is already updated is not further updated + // to prevent a new status update from being broadcast + let live_bids = auction_service + .get_live_bids(GetLiveBidsInput { + permission_key: permission_key_svm, + }) + .await; + if !live_bids + .iter() + .any(|bid| bid.id == winner_bid.id && bid.status.is_submitted()) + { + tracing::error!(winner_bid = ?winner_bid, opportunity = ?opportunity, "Failed to update winner bid status"); + return Err(RestError::TemporarilyUnavailable); + } Ok(entities::Quote { - transaction: bid.chain_data.transaction.clone(), - expiration_time: submit_bid_data.deadline, - input_token: input.quote_create.input_token, - output_token: entities::TokenAmountSvm { - token: input.quote_create.output_mint_token, - amount: output_amount, - }, - maximum_slippage_percentage: input.quote_create.maximum_slippage_percentage, - chain_id: input.quote_create.chain_id, + transaction: bid.chain_data.transaction.clone(), + expiration_time: deadline, + input_token, + output_token, // TODO*: incorporate fees (when fees are in the output token) + chain_id: input.quote_create.chain_id, }) } } diff --git a/auction-server/src/opportunity/service/mod.rs b/auction-server/src/opportunity/service/mod.rs index 06e600c4..fa3852ab 100644 --- a/auction-server/src/opportunity/service/mod.rs +++ b/auction-server/src/opportunity/service/mod.rs @@ -31,13 +31,11 @@ use { types::Address, }, futures::future::try_join_all, - solana_sdk::pubkey::Pubkey, std::{ collections::HashMap, sync::Arc, }, tokio::sync::RwLock, - tokio_util::task::TaskTracker, }; pub mod add_opportunity; @@ -50,7 +48,6 @@ pub mod remove_invalid_or_expired_opportunities; pub mod remove_opportunities; pub mod verification; -mod estimate_price; mod get_spoof_info; mod make_adapter_calldata; mod make_opportunity_execution_params; @@ -84,8 +81,7 @@ impl ConfigEvm { // NOTE: Do not implement debug here. it has a circular reference to auction_service pub struct ConfigSvm { - pub wallet_program_router_account: Pubkey, - pub auction_service: RwLock>>, + pub auction_service: RwLock>>, } impl ConfigSvm { @@ -198,12 +194,11 @@ impl ConfigSvm { ) -> anyhow::Result> { Ok(chains .iter() - .map(|(chain_id, config)| { + .map(|(chain_id, _)| { ( chain_id.clone(), Self { - wallet_program_router_account: config.wallet_program_router_account, - auction_service: RwLock::new(None), + auction_service: RwLock::new(None), }, ) }) @@ -241,18 +236,16 @@ impl ChainType for ChainTypeSvm { // TODO maybe just create a service per chain_id? pub struct Service { - store: Arc, - db: DB, + store: Arc, + db: DB, // TODO maybe after adding state for opportunity we can remove the arc - repo: Arc>, - config: HashMap, - task_tracker: TaskTracker, + repo: Arc>, + config: HashMap, } impl Service { pub fn new(store: Arc, db: DB, config: HashMap) -> Self { Self { - task_tracker: store.task_tracker.clone(), store, db, repo: Arc::new(Repository::::new()), diff --git a/auction-server/src/server.rs b/auction-server/src/server.rs index 6fe2c9c0..910f6a7b 100644 --- a/auction-server/src/server.rs +++ b/auction-server/src/server.rs @@ -251,7 +251,6 @@ pub async fn start_server(run_options: RunOptions) -> Result<()> { broadcast_sender, broadcast_receiver, }, - task_tracker: task_tracker.clone(), secret_key: run_options.secret_key.clone(), access_tokens: RwLock::new(access_tokens), metrics_recorder: setup_metrics_recorder()?, @@ -324,25 +323,45 @@ pub async fn start_server(run_options: RunOptions) -> Result<()> { .config .wallet_program_router_account, express_relay: auction_service::ExpressRelaySvm { - program_id: chain_store + program_id: chain_store .config .express_relay_program_id, - relayer: Keypair::from_base58_string( + relayer: Keypair::from_base58_string( &run_options .private_key_svm .clone() .expect("No svm private key provided for chain"), ), - permission_account_position: env!( + permission_account_position_submit_bid: env!( "SUBMIT_BID_PERMISSION_ACCOUNT_POSITION" ) .parse::() .expect("Failed to parse permission account position"), - router_account_position: env!( + router_account_position_submit_bid: env!( "SUBMIT_BID_ROUTER_ACCOUNT_POSITION" ) .parse::() - .expect("Failed to parse router account position"), + .expect("Failed to parse (submit bid) router account position"), + router_account_position_swap: env!( + "SWAP_ROUTER_ACCOUNT_POSITION" + ) + .parse::() + .expect("Failed to parse (swap) router account position"), + user_wallet_account_position_swap: env!( + "SWAP_USER_WALLET_ACCOUNT_POSITION" + ) + .parse::() + .expect("Failed to parse user wallet account position"), + mint_input_account_position_swap: env!( + "SWAP_MINT_INPUT_ACCOUNT_POSITION" + ) + .parse::() + .expect("Failed to parse user wallet account position"), + mint_output_account_position_swap: env!( + "SWAP_MINT_OUTPUT_ACCOUNT_POSITION" + ) + .parse::() + .expect("Failed to parse user wallet account position"), }, ws_address: chain_store.config.ws_addr.clone(), tx_broadcaster_client: TracedSenderSvm::new_client( diff --git a/auction-server/src/state.rs b/auction-server/src/state.rs index 1469e1d1..30a2bd44 100644 --- a/auction-server/src/state.rs +++ b/auction-server/src/state.rs @@ -31,7 +31,6 @@ use { Response, RpcLogsResponse, }, - solana_sdk::pubkey::Pubkey, std::{ collections::HashMap, sync::Arc, @@ -45,7 +44,6 @@ use { }, RwLock, }, - tokio_util::task::TaskTracker, uuid::Uuid, }; @@ -105,11 +103,10 @@ impl ChainStoreEvm { } pub struct ChainStoreSvm { - pub log_sender: Sender>, + pub log_sender: Sender>, // only to avoid closing the channel - pub _dummy_log_receiver: Receiver>, - pub config: ConfigSvm, - pub wallet_program_router_account: Pubkey, + pub _dummy_log_receiver: Receiver>, + pub config: ConfigSvm, } impl ChainStoreSvm { @@ -119,8 +116,6 @@ impl ChainStoreSvm { Self { log_sender: tx, _dummy_log_receiver: rx, - - wallet_program_router_account: config.wallet_program_router_account, config, } } @@ -131,7 +126,6 @@ pub struct Store { pub chains_svm: HashMap>, pub ws: WsState, pub db: sqlx::PgPool, - pub task_tracker: TaskTracker, pub secret_key: String, pub access_tokens: RwLock>, pub metrics_recorder: PrometheusHandle,