From 6c13403c4fc5be47d8ba4b7f9e83ba527d2d8235 Mon Sep 17 00:00:00 2001 From: 0xNe0x1 <0xneo11@gmail.com> Date: Wed, 25 Dec 2024 06:29:00 +0100 Subject: [PATCH] adds initial raydium support (#22) * adds initial raydium support * fix lint --- dev.html | 6 +- dist/esm/index.evm.js | 5 +- dist/esm/index.js | 1039 +++++++++++++++++-- dist/esm/index.solana.js | 967 +++++++++++++++-- dist/umd/index.evm.js | 5 +- dist/umd/index.js | 1037 ++++++++++++++++-- dist/umd/index.solana.js | 965 +++++++++++++++-- examples/raydium.md | 20 + package.evm.json | 2 +- package.json | 2 +- package.solana.json | 2 +- src/classes/Exchange.js | 5 +- src/exchanges.js | 6 + src/exchanges/raydium.js | 48 + src/platforms/solana/orca.js | 12 - src/platforms/solana/orca/amounts.js | 16 +- src/platforms/solana/orca/index.js | 12 + src/platforms/solana/orca/pairs.js | 1 - src/platforms/solana/raydium/amm/pairs.js | 136 +++ src/platforms/solana/raydium/amounts.js | 101 ++ src/platforms/solana/raydium/clmm/pairs.js | 7 + src/platforms/solana/raydium/cpamm/pairs.js | 7 + src/platforms/solana/raydium/index.js | 51 + src/platforms/solana/raydium/layouts.js | 176 ++++ src/platforms/solana/raydium/pairs.js | 64 ++ src/platforms/solana/raydium/path.js | 91 ++ src/platforms/solana/raydium/transaction.js | 307 ++++++ 27 files changed, 4682 insertions(+), 408 deletions(-) create mode 100644 examples/raydium.md create mode 100644 src/exchanges/raydium.js delete mode 100644 src/platforms/solana/orca.js create mode 100644 src/platforms/solana/orca/index.js create mode 100644 src/platforms/solana/raydium/amm/pairs.js create mode 100644 src/platforms/solana/raydium/amounts.js create mode 100644 src/platforms/solana/raydium/clmm/pairs.js create mode 100644 src/platforms/solana/raydium/cpamm/pairs.js create mode 100644 src/platforms/solana/raydium/index.js create mode 100644 src/platforms/solana/raydium/layouts.js create mode 100644 src/platforms/solana/raydium/pairs.js create mode 100644 src/platforms/solana/raydium/path.js create mode 100644 src/platforms/solana/raydium/transaction.js diff --git a/dev.html b/dev.html index 71c7c0a..64ba790 100644 --- a/dev.html +++ b/dev.html @@ -8,10 +8,10 @@ - + - - + + diff --git a/dist/esm/index.evm.js b/dist/esm/index.evm.js index 68ffa77..255e09a 100644 --- a/dist/esm/index.evm.js +++ b/dist/esm/index.evm.js @@ -306,7 +306,10 @@ const route$1 = ({ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { diff --git a/dist/esm/index.js b/dist/esm/index.js index ab539dd..f8da0d4 100644 --- a/dist/esm/index.js +++ b/dist/esm/index.js @@ -2,10 +2,10 @@ import Token from '@depay/web3-tokens'; import { request, getProvider } from '@depay/web3-client'; import { ethers } from 'ethers'; import Blockchains from '@depay/web3-blockchains'; -import { BN, struct, publicKey, u128, u64 as u64$1, seq, u8, u16, i32, bool, i128, PublicKey, Buffer, Keypair, SystemProgram, TransactionInstruction } from '@depay/solana-web3.js'; +import { BN, struct, publicKey, u128, u64 as u64$1, seq, u8, u16, i32, bool, i128, PublicKey, Buffer, Keypair, SystemProgram, TransactionInstruction, blob, u32 } from '@depay/solana-web3.js'; import Decimal from 'decimal.js'; -function _optionalChain$5(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { +function _optionalChain$7(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { constructor({ blockchain, tokenIn, @@ -31,10 +31,10 @@ function _optionalChain$5(ops) { let lastAccessLHS = undefined; let value = ops[ this.decimalsOut = decimalsOut; this.path = path; this.pools = pools; - this.amountIn = _optionalChain$5([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); - this.amountOutMin = _optionalChain$5([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); - this.amountOut = _optionalChain$5([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); - this.amountInMax = _optionalChain$5([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); + this.amountIn = _optionalChain$7([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); + this.amountOutMin = _optionalChain$7([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); + this.amountOut = _optionalChain$7([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); + this.amountInMax = _optionalChain$7([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); this.exchange = exchange; this.getPrep = getPrep; this.getTransaction = getTransaction; @@ -308,7 +308,10 @@ const route$1 = ({ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { @@ -432,7 +435,7 @@ class Exchange { } } -function _optionalChain$4(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } +function _optionalChain$6(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } // Replaces 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE with the wrapped token and implies wrapping. // @@ -440,7 +443,7 @@ function _optionalChain$4(ops) { let lastAccessLHS = undefined; let value = ops[ // to be able to differentiate between ETH<>Token and WETH<>Token swaps // as they are not the same! // -const getExchangePath$4 = ({ blockchain, exchange, path }) => { +const getExchangePath$5 = ({ blockchain, exchange, path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( @@ -472,8 +475,8 @@ const minReserveRequirements = ({ reserves, min, token, token0, token1, decimals } }; -const pathExists$5 = async ({ blockchain, exchange, path }) => { - const exchangePath = getExchangePath$4({ blockchain, exchange, path }); +const pathExists$6 = async ({ blockchain, exchange, path }) => { + const exchangePath = getExchangePath$5({ blockchain, exchange, path }); if(!exchangePath || exchangePath.length === 1) { return false } try { let pair = await request({ @@ -482,7 +485,7 @@ const pathExists$5 = async ({ blockchain, exchange, path }) => { method: 'getPair', api: exchange[blockchain].factory.api, cache: 3600000, - params: getExchangePath$4({ blockchain, exchange, path }), + params: getExchangePath$5({ blockchain, exchange, path }), }); if(!pair || pair == Blockchains[blockchain].zero) { return false } let [reserves, token0, token1] = await Promise.all([ @@ -503,61 +506,61 @@ const pathExists$5 = async ({ blockchain, exchange, path }) => { } catch (e){ console.log('e', e); return false } }; -const findPath$5 = async ({ blockchain, exchange, tokenIn, tokenOut }) => { +const findPath$6 = async ({ blockchain, exchange, tokenIn, tokenOut }) => { if( [tokenIn, tokenOut].includes(Blockchains[blockchain].currency.address) && [tokenIn, tokenOut].includes(Blockchains[blockchain].wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path; - if (await pathExists$5({ blockchain, exchange, path: [tokenIn, tokenOut] })) { + if (await pathExists$6({ blockchain, exchange, path: [tokenIn, tokenOut] })) { // direct path path = [tokenIn, tokenOut]; } else if ( tokenIn != Blockchains[blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenIn, Blockchains[blockchain].wrapped.address] }) && + await pathExists$6({ blockchain, exchange, path: [tokenIn, Blockchains[blockchain].wrapped.address] }) && tokenOut != Blockchains[blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenOut, Blockchains[blockchain].wrapped.address] }) + await pathExists$6({ blockchain, exchange, path: [tokenOut, Blockchains[blockchain].wrapped.address] }) ) { // path via WRAPPED path = [tokenIn, Blockchains[blockchain].wrapped.address, tokenOut]; } else if ( !Blockchains[blockchain].stables.usd.includes(tokenIn) && - (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [tokenIn, stable] })))).filter(Boolean).length && + (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [tokenIn, stable] })))).filter(Boolean).length && tokenOut != Blockchains[blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [Blockchains[blockchain].wrapped.address, tokenOut] }) + await pathExists$6({ blockchain, exchange, path: [Blockchains[blockchain].wrapped.address, tokenOut] }) ) { // path via tokenIn -> USD -> WRAPPED -> tokenOut - let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [tokenIn, stable] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [tokenIn, stable] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, USD, Blockchains[blockchain].wrapped.address, tokenOut]; } else if ( tokenIn != Blockchains[blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenIn, Blockchains[blockchain].wrapped.address] }) && + await pathExists$6({ blockchain, exchange, path: [tokenIn, Blockchains[blockchain].wrapped.address] }) && !Blockchains[blockchain].stables.usd.includes(tokenOut) && - (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length + (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length ) { // path via tokenIn -> WRAPPED -> USD -> tokenOut - let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, Blockchains[blockchain].wrapped.address, USD, tokenOut]; } else if ( tokenIn != Blockchains[blockchain].wrapped.address && tokenIn === Blockchains[blockchain].currency.address && !Blockchains[blockchain].stables.usd.includes(tokenOut) && - (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length + (await Promise.all(Blockchains[blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length ) { - let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains[blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, USD, tokenOut]; } // Add WRAPPED to route path if things start or end with NATIVE // because that actually reflects how things are routed in reality: - if(_optionalChain$4([path, 'optionalAccess', _ => _.length]) && path[0] == Blockchains[blockchain].currency.address) { + if(_optionalChain$6([path, 'optionalAccess', _ => _.length]) && path[0] == Blockchains[blockchain].currency.address) { path.splice(1, 0, Blockchains[blockchain].wrapped.address); - } else if(_optionalChain$4([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == Blockchains[blockchain].currency.address) { + } else if(_optionalChain$6([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == Blockchains[blockchain].currency.address) { path.splice(path.length-1, 0, Blockchains[blockchain].wrapped.address); } - return { path, exchangePath: getExchangePath$4({ blockchain, exchange, path }) } + return { path, exchangePath: getExchangePath$5({ blockchain, exchange, path }) } }; let getAmountOut$3 = ({ blockchain, exchange, path, amountIn, tokenIn, tokenOut }) => { @@ -569,7 +572,7 @@ let getAmountOut$3 = ({ blockchain, exchange, path, amountIn, tokenIn, tokenOut api: exchange[blockchain].router.api, params: { amountIn: amountIn, - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), }, }) .then((amountsOut)=>{ @@ -588,7 +591,7 @@ let getAmountIn$3 = ({ blockchain, exchange, path, amountOut, block }) => { api: exchange[blockchain].router.api, params: { amountOut: amountOut, - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), }, block }) @@ -597,7 +600,7 @@ let getAmountIn$3 = ({ blockchain, exchange, path, amountOut, block }) => { }) }; -let getAmounts$5 = async ({ +let getAmounts$6 = async ({ blockchain, exchange, path, @@ -674,7 +677,7 @@ let getPrep$3 = async({ }; -let getTransaction$5 = ({ +let getTransaction$6 = ({ exchange, blockchain, path, @@ -725,7 +728,7 @@ let getTransaction$5 = ({ } transaction.params = Object.assign({}, transaction.params, { - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), to: account, deadline: Math.round(Date.now() / 1000) + 60 * 60 * 24, // 24 hours }); @@ -738,17 +741,17 @@ const FACTORY$3 = [{"inputs":[{"internalType":"address","name":"_feeToSetter","t const PAIR$1 = [{"inputs":[],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1","type":"uint256"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"Burn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0In","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1In","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount0Out","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1Out","type":"uint256"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint112","name":"reserve0","type":"uint112"},{"indexed":false,"internalType":"uint112","name":"reserve1","type":"uint112"}],"name":"Sync","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"constant":true,"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"MINIMUM_LIQUIDITY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"PERMIT_TYPEHASH","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"burn","outputs":[{"internalType":"uint256","name":"amount0","type":"uint256"},{"internalType":"uint256","name":"amount1","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"factory","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"getReserves","outputs":[{"internalType":"uint112","name":"_reserve0","type":"uint112"},{"internalType":"uint112","name":"_reserve1","type":"uint112"},{"internalType":"uint32","name":"_blockTimestampLast","type":"uint32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token0","type":"address"},{"internalType":"address","name":"_token1","type":"address"}],"name":"initialize","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"kLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"mint","outputs":[{"internalType":"uint256","name":"liquidity","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"price0CumulativeLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"price1CumulativeLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"skim","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"uint256","name":"amount0Out","type":"uint256"},{"internalType":"uint256","name":"amount1Out","type":"uint256"},{"internalType":"address","name":"to","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"swap","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"sync","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"token0","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"token1","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]; var UniswapV2 = { - findPath: findPath$5, - pathExists: pathExists$5, - getAmounts: getAmounts$5, + findPath: findPath$6, + pathExists: pathExists$6, + getAmounts: getAmounts$6, getPrep: getPrep$3, - getTransaction: getTransaction$5, + getTransaction: getTransaction$6, ROUTER: ROUTER$3, FACTORY: FACTORY$3, PAIR: PAIR$1, }; -const exchange$h = { +const exchange$i = { name: 'honeyswap', label: 'Honeyswap', @@ -778,13 +781,13 @@ var honeyswap = (scope)=>{ return new Exchange( - Object.assign(exchange$h, { + Object.assign(exchange$i, { scope, - findPath: (args)=>UniswapV2.findPath({ ...args, exchange: exchange$h }), - pathExists: (args)=>UniswapV2.pathExists({ ...args, exchange: exchange$h }), - getAmounts: (args)=>UniswapV2.getAmounts({ ...args, exchange: exchange$h }), - getPrep: (args)=>UniswapV2.getPrep({ ...args, exchange: exchange$h }), - getTransaction: (args)=>UniswapV2.getTransaction({ ...args, exchange: exchange$h }), + findPath: (args)=>UniswapV2.findPath({ ...args, exchange: exchange$i }), + pathExists: (args)=>UniswapV2.pathExists({ ...args, exchange: exchange$i }), + getAmounts: (args)=>UniswapV2.getAmounts({ ...args, exchange: exchange$i }), + getPrep: (args)=>UniswapV2.getPrep({ ...args, exchange: exchange$i }), + getTransaction: (args)=>UniswapV2.getTransaction({ ...args, exchange: exchange$i }), }) ) }; @@ -1944,7 +1947,7 @@ let getAccounts = async (base, quote) => { return accounts }; -let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { +let getPairsWithPrice$4 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { try { let accounts = await getAccounts(tokenIn, tokenOut); if(accounts.length === 0) { accounts = await getAccounts(tokenOut, tokenIn); } @@ -1971,32 +1974,31 @@ let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amount } }; -let getHighestPrice = (pairs)=>{ +let getHighestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).gt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; -let getLowestPrice = (pairs)=>{ +let getLowestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).lt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; -let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { - const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); +let getBestPair$1 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice$4({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); if(!pairs || pairs.length === 0) { return } let bestPair; if(amountIn || amountInMax) { - bestPair = getHighestPrice(pairs); + bestPair = getHighestPrice$1(pairs); } else { // amount out - bestPair = getLowestPrice(pairs); + bestPair = getLowestPrice$1(pairs); } - return bestPair }; -function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } -const blockchain$1 = Blockchains.solana; +function _optionalChain$5(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } +const blockchain$3 = Blockchains.solana; // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. // @@ -2004,64 +2006,64 @@ const blockchain$1 = Blockchains.solana; // to be able to differentiate between SOL<>Token and WSOL<>Token swaps // as they are not the same! // -let getExchangePath$3 = ({ path }) => { +let getExchangePath$4 = ({ path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( - token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && - path[index-1] != blockchain$1.wrapped.address + token === blockchain$3.currency.address && path[index+1] != blockchain$3.wrapped.address && + path[index-1] != blockchain$3.wrapped.address ) { - return blockchain$1.wrapped.address + return blockchain$3.wrapped.address } else { return token } }); - if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + if(exchangePath[0] == blockchain$3.currency.address && exchangePath[1] == blockchain$3.wrapped.address) { exchangePath.splice(0, 1); - } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + } else if(exchangePath[exchangePath.length-1] == blockchain$3.currency.address && exchangePath[exchangePath.length-2] == blockchain$3.wrapped.address) { exchangePath.splice(exchangePath.length-1, 1); } return exchangePath }; -let pathExists$4 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { +let pathExists$5 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { if(path.length == 1) { return false } - path = getExchangePath$3({ path }); - if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + path = getExchangePath$4({ path }); + if((await getPairsWithPrice$4({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { return true } else { return false } }; -let findPath$4 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { +let findPath$5 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { if( - [tokenIn, tokenOut].includes(blockchain$1.currency.address) && - [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + [tokenIn, tokenOut].includes(blockchain$3.currency.address) && + [tokenIn, tokenOut].includes(blockchain$3.wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path, stablesIn, stablesOut, stable; - if (await pathExists$4({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + if (await pathExists$5({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { // direct path path = [tokenIn, tokenOut]; } else if ( - tokenIn != blockchain$1.wrapped.address && - tokenIn != blockchain$1.currency.address && - await pathExists$4({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && - tokenOut != blockchain$1.wrapped.address && - tokenOut != blockchain$1.currency.address && - await pathExists$4({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + tokenIn != blockchain$3.wrapped.address && + tokenIn != blockchain$3.currency.address && + await pathExists$5({ path: [tokenIn, blockchain$3.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$3.wrapped.address && + tokenOut != blockchain$3.currency.address && + await pathExists$5({ path: [tokenOut, blockchain$3.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ) { // path via blockchain.wrapped.address - path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + path = [tokenIn, blockchain$3.wrapped.address, tokenOut]; } else if ( - !blockchain$1.stables.usd.includes(tokenIn) && - (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && - !blockchain$1.stables.usd.includes(tokenOut) && - (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$5({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$5({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) ) { // path via TOKEN_IN <> STABLE <> TOKEN_OUT @@ -2070,22 +2072,26 @@ let findPath$4 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, a // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address // because that actually reflects how things are routed in reality: - if(_optionalChain$3([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { - path.splice(1, 0, blockchain$1.wrapped.address); - } else if(_optionalChain$3([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { - path.splice(path.length-1, 0, blockchain$1.wrapped.address); + if(_optionalChain$5([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$3.currency.address) { + path.splice(1, 0, blockchain$3.wrapped.address); + } else if(_optionalChain$5([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$3.currency.address) { + path.splice(path.length-1, 0, blockchain$3.wrapped.address); } - return { path, exchangePath: getExchangePath$3({ path }) } + return { path, exchangePath: getExchangePath$4({ path }) } }; -let getAmountsOut = async ({ path, amountIn, amountInMax }) => { +let getAmountsOut$1 = async ({ path, amountIn, amountInMax }) => { let amounts = [ethers.BigNumber.from(amountIn || amountInMax)]; - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -2093,15 +2099,19 @@ let getAmountsOut = async ({ path, amountIn, amountInMax }) => { return amounts }; -let getAmountsIn = async({ path, amountOut, amountOutMin }) => { +let getAmountsIn$1 = async({ path, amountOut, amountOutMin }) => { path = path.slice().reverse(); let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)]; - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -2109,7 +2119,7 @@ let getAmountsIn = async({ path, amountOut, amountOutMin }) => { return amounts.slice().reverse() }; -let getAmounts$4 = async ({ +let getAmounts$5 = async ({ path, tokenIn, tokenOut, @@ -2118,10 +2128,10 @@ let getAmounts$4 = async ({ amountInMax, amountOutMin }) => { - path = getExchangePath$3({ path }); + path = getExchangePath$4({ path }); let amounts; if (amountOut) { - amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOut, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -2129,7 +2139,7 @@ let getAmounts$4 = async ({ amountInMax = amountIn; } } else if (amountIn) { - amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountIn, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -2137,7 +2147,7 @@ let getAmounts$4 = async ({ amountOutMin = amountOut; } } else if(amountOutMin) { - amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOutMin, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -2145,7 +2155,7 @@ let getAmounts$4 = async ({ amountInMax = amountIn; } } else if(amountInMax) { - amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountInMax, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -2162,7 +2172,7 @@ let getAmounts$4 = async ({ } }; -const blockchain = Blockchains.solana; +const blockchain$2 = Blockchains.solana; const SWAP_INSTRUCTION = new BN("14449647541112719096"); const TWO_HOP_SWAP_INSTRUCTION = new BN("16635068063392030915"); @@ -2377,7 +2387,7 @@ const getSwapInstructionData = ({ amount, otherAmountThreshold, sqrtPriceLimit, return data }; -const getTransaction$4 = async ({ +const getTransaction$5 = async ({ path, amountIn, amountInMax, @@ -2393,7 +2403,7 @@ const getTransaction$4 = async ({ let transaction = { blockchain: 'solana' }; let instructions = []; - const exchangePath = getExchangePath$3({ path }); + const exchangePath = getExchangePath$4({ path }); if(exchangePath.length > 3) { throw 'Orca can only handle fixed paths with a max length of 3 (2 pools)!' } const tokenIn = exchangePath[0]; const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; @@ -2401,19 +2411,19 @@ const getTransaction$4 = async ({ let pairs; if(exchangePath.length == 2) { - pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + pairs = [await getBestPair$1({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; } else { if(amountInInput || amountInMaxInput) { - pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; - pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); } else { // originally amountOut - pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; - pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); } } - let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); - let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let startsWrapped = (path[0] === blockchain$2.currency.address && exchangePath[0] === blockchain$2.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain$2.currency.address && exchangePath[exchangePath.length-1] === blockchain$2.wrapped.address); let wrappedAccount; const provider = await getProvider('solana'); @@ -2434,7 +2444,7 @@ const getTransaction$4 = async ({ instructions.push( Token.solana.initializeAccountInstruction({ account: wrappedAccount, - token: blockchain.wrapped.address, + token: blockchain$2.wrapped.address, owner: account }) ); @@ -2531,14 +2541,14 @@ const getTransaction$4 = async ({ }; var Orca = { - findPath: findPath$4, - pathExists: pathExists$4, - getAmounts: getAmounts$4, - getTransaction: getTransaction$4, + findPath: findPath$5, + pathExists: pathExists$5, + getAmounts: getAmounts$5, + getTransaction: getTransaction$5, WHIRLPOOL_LAYOUT, }; -const exchange$g = { +const exchange$h = { name: 'orca', label: 'Orca', @@ -2559,16 +2569,815 @@ const exchange$g = { var orca = (scope)=>{ + return new Exchange( + + Object.assign(exchange$h, { + scope, + + findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$h }), + pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$h }), + getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$h }), + getPrep: (args)=>{}, + getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$h }), + }) + ) +}; + +// OpenBook Market +const MARKET_LAYOUT = struct([ + blob(5), + blob(8), // accountFlagsLayout('accountFlags'), + publicKey('ownAddress'), + u64$1('vaultSignerNonce'), + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('baseVault'), + u64$1('baseDepositsTotal'), + u64$1('baseFeesAccrued'), + publicKey('quoteVault'), + u64$1('quoteDepositsTotal'), + u64$1('quoteFeesAccrued'), + u64$1('quoteDustThreshold'), + publicKey('requestQueue'), + publicKey('eventQueue'), + publicKey('bids'), + publicKey('asks'), + u64$1('baseLotSize'), + u64$1('quoteLotSize'), + u64$1('feeRateBps'), + u64$1('referrerRebatesAccrued'), + blob(7), +]); + +// AMM 675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8 +const AMM_LAYOUT = struct([ + u64$1('status'), + u64$1('nonce'), + u64$1('maxOrder'), + u64$1('depth'), + u64$1('baseDecimal'), + u64$1('quoteDecimal'), + u64$1('state'), + u64$1('resetFlag'), + u64$1('minSize'), + u64$1('volMaxCutRatio'), + u64$1('amountWaveRatio'), + u64$1('baseLotSize'), + u64$1('quoteLotSize'), + u64$1('minPriceMultiplier'), + u64$1('maxPriceMultiplier'), + u64$1('systemDecimalValue'), + u64$1('minSeparateNumerator'), + u64$1('minSeparateDenominator'), + u64$1('tradeFeeNumerator'), + u64$1('tradeFeeDenominator'), + u64$1('pnlNumerator'), + u64$1('pnlDenominator'), + u64$1('swapFeeNumerator'), + u64$1('swapFeeDenominator'), + u64$1('baseNeedTakePnl'), + u64$1('quoteNeedTakePnl'), + u64$1('quoteTotalPnl'), + u64$1('baseTotalPnl'), + u64$1('poolOpenTime'), + u64$1('punishPcAmount'), + u64$1('punishCoinAmount'), + u64$1('orderbookToInitTime'), + u128('swapBaseInAmount'), + u128('swapQuoteOutAmount'), + u64$1('swapBase2QuoteFee'), + u128('swapQuoteInAmount'), + u128('swapBaseOutAmount'), + u64$1('swapQuote2BaseFee'), + // amm vault + publicKey('baseVault'), + publicKey('quoteVault'), + // mint + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('lpMint'), + // market + publicKey('openOrders'), + publicKey('marketId'), + publicKey('marketProgramId'), + publicKey('targetOrders'), + publicKey('withdrawQueue'), + publicKey('lpVault'), + publicKey('owner'), + // true circulating supply without lock up + u64$1('lpReserve'), + seq(u64$1(), 3, 'padding'), +]); + +// CP_AMM CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C +const CPAMM_LAYOUT = struct([ + blob(8), + publicKey("configId"), + publicKey("poolCreator"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("mintLp"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("mintProgramA"), + publicKey("mintProgramB"), + publicKey("observationId"), + u8("bump"), + u8("status"), + u8("lpDecimals"), + u8("mintDecimalA"), + u8("mintDecimalB"), + u64$1("lpAmount"), + u64$1("protocolFeesMintA"), + u64$1("protocolFeesMintB"), + u64$1("fundFeesMintA"), + u64$1("fundFeesMintB"), + u64$1("openTime"), + seq(u64$1(), 32), +]); + +// CLMM + +const RewardInfo = struct([ + u8("rewardState"), + u64$1("openTime"), + u64$1("endTime"), + u64$1("lastUpdateTime"), + u128("emissionsPerSecondX64"), + u64$1("rewardTotalEmissioned"), + u64$1("rewardClaimed"), + publicKey("tokenMint"), + publicKey("tokenVault"), + publicKey("creator"), + u128("rewardGrowthGlobalX64"), +]); + +const CLMM_LAYOUT = struct([ + blob(8), + u8("bump"), + publicKey("ammConfig"), + publicKey("creator"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("observationId"), + u8("mintDecimalsA"), + u8("mintDecimalsB"), + u16("tickSpacing"), + u128("liquidity"), + u128("sqrtPriceX64"), + i32("tickCurrent"), + u32(), + u128("feeGrowthGlobalX64A"), + u128("feeGrowthGlobalX64B"), + u64$1("protocolFeesTokenA"), + u64$1("protocolFeesTokenB"), + u128("swapInAmountTokenA"), + u128("swapOutAmountTokenB"), + u128("swapInAmountTokenB"), + u128("swapOutAmountTokenA"), + u8("status"), + seq(u8(), 7, ""), + seq(RewardInfo, 3, "rewardInfos"), + seq(u64$1(), 16, "tickArrayBitmap"), + u64$1("totalFeesTokenA"), + u64$1("totalFeesClaimedTokenA"), + u64$1("totalFeesTokenB"), + u64$1("totalFeesClaimedTokenB"), + u64$1("fundFeesTokenA"), + u64$1("fundFeesTokenB"), + u64$1("startTime"), + seq(u64$1(), 15 * 4 - 3, "padding"), +]); + +function _optionalChain$4(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + +const LIQUIDITY_FEES_NUMERATOR = new BN(25); +const LIQUIDITY_FEES_DENOMINATOR = new BN(10000); + +const BNDivCeil = (bn1, bn2)=> { + const { div, mod } = bn1.divmod(bn2); + + if (mod.gt(new BN(0))) { + return div.add(new BN(1)) + } else { + return div + } +}; + +const getPairs = (base, quote)=>{ + return request(`solana://675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8/getProgramAccounts`, { + params: { filters: [ + { dataSize: AMM_LAYOUT.span }, + { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 (Swap) + { memcmp: { offset: 400, bytes: base }}, + { memcmp: { offset: 432, bytes: quote }}, + ]}, + api: AMM_LAYOUT, + cache: 86400, // 24h, + cacheKey: ['raydium', base.toString(), quote.toString()].join('-') + }) +}; + +const getPairsWithPrice$3 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + let accounts = await getPairs(tokenIn, tokenOut); + + if(accounts.length == 0) { + accounts = await getPairs(tokenOut, tokenIn); + } + + const pairs = await Promise.all(accounts.map(async(account)=>{ + + const baseMint = account.data.baseMint.toString(); + const quoteMint = account.data.quoteMint.toString(); + + const balances = await Promise.all([ + request(`solana://${account.data.baseVault.toString()}/getTokenAccountBalance`, { cache: 5 }), + request(`solana://${account.data.quoteVault.toString()}/getTokenAccountBalance`, { cache: 5 }) + ]); + const baseReserve = (_optionalChain$4([balances, 'access', _ => _[0], 'optionalAccess', _2 => _2.value, 'optionalAccess', _3 => _3.amount]) ? new BN(_optionalChain$4([balances, 'access', _4 => _4[0], 'optionalAccess', _5 => _5.value, 'optionalAccess', _6 => _6.amount])) : new BN(0)).sub(account.data.baseNeedTakePnl); + const quoteReserve = (_optionalChain$4([balances, 'access', _7 => _7[1], 'optionalAccess', _8 => _8.value, 'optionalAccess', _9 => _9.amount]) ? new BN(_optionalChain$4([balances, 'access', _10 => _10[1], 'optionalAccess', _11 => _11.value, 'optionalAccess', _12 => _12.amount])) : new BN(0)).sub(account.data.quoteNeedTakePnl); + + if(baseMint === Blockchains.solana.wrapped.address) { + if(baseReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (quoteMint === Blockchains.solana.wrapped.address) { + if(quoteReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(baseMint)) { + if((parseFloat(baseReserve.toString()) / 10 ** account.data.baseDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(quoteMint)) { + if((parseFloat(quoteReserve.toString()) / 10 ** account.data.quoteDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } + + const reserves = [baseReserve, quoteReserve]; + + if(tokenOut === baseMint) { + reserves.reverse(); + } + + const [reserveIn, reserveOut] = reserves; + + let price; + if(amountOut || amountOutMin) { // compute amountIn + + new BN(0); + let amountOutRaw = new BN((amountOut || amountOutMin).toString()); + + if (amountOutRaw.gt(reserveOut)) { + amountOutRaw = reserveOut.sub(new BN(1)); + } + + const denominator = reserveOut.sub(amountOutRaw); + const amountInWithoutFee = reserveIn.mul(amountOutRaw).div(denominator); + + price = amountInWithoutFee + .mul(LIQUIDITY_FEES_DENOMINATOR) + .div(LIQUIDITY_FEES_DENOMINATOR.sub(LIQUIDITY_FEES_NUMERATOR)) + .toString(); + + } else { // compute amountOut + + new BN(0); + const amountInRaw = new BN((amountIn || amountInMin).toString()); + const feeRaw = BNDivCeil(amountInRaw.mul(LIQUIDITY_FEES_NUMERATOR), LIQUIDITY_FEES_DENOMINATOR); + + const amountInWithFee = amountInRaw.sub(feeRaw); + const denominator = reserveIn.add(amountInWithFee); + + price = reserveOut.mul(amountInWithFee).div(denominator).toString(); + } + + return { + publicKey: account.pubkey.toString(), + baseMint, + quoteMint, + price, + data: account.data + } + + })); + + return pairs.filter(Boolean) +}; + +const getPairsWithPrice$2 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +}; + +const getPairsWithPrice$1 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +}; + +const getParisWithPriceForAllTypes = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + return Promise.all([ + getPairsWithPrice$3({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$2({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$1({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + ]).then((pairsAMM, pairsCPAMM, pairsCLMNN)=>{ + return [ + (pairsAMM || []).filter(Boolean).flat(), + (pairsCPAMM || []).filter(Boolean).flat(), + (pairsCLMNN || []).filter(Boolean).flat() + ].flat() + }) +}; + +const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + try { + return await getParisWithPriceForAllTypes({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + } catch (e) { + return [] + } +}; + +let getHighestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).gt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +}; + +let getLowestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).lt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +}; + +let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + + if(!pairs || pairs.length === 0) { return } + + let bestPair; + + if(amountIn || amountInMax) { + bestPair = getHighestPrice(pairs); + } else { // amount out + bestPair = getLowestPrice(pairs); + } + + return bestPair +}; + +function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } +const blockchain$1 = Blockchains.solana; + +// Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. +// +// We keep 11111111111111111111111111111111 internally +// to be able to differentiate between SOL<>Token and WSOL<>Token swaps +// as they are not the same! +// +let getExchangePath$3 = ({ path }) => { + if(!path) { return } + let exchangePath = path.map((token, index) => { + if ( + token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && + path[index-1] != blockchain$1.wrapped.address + ) { + return blockchain$1.wrapped.address + } else { + return token + } + }); + + if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + exchangePath.splice(0, 1); + } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + exchangePath.splice(exchangePath.length-1, 1); + } + + return exchangePath +}; + +let pathExists$4 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + if(path.length == 1) { return false } + path = getExchangePath$3({ path }); + if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + return true + } else { + return false + } +}; + +let findPath$4 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + if( + [tokenIn, tokenOut].includes(blockchain$1.currency.address) && + [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + ) { return { path: undefined, exchangePath: undefined } } + + let path, stablesIn, stablesOut, stable; + + if (await pathExists$4({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + // direct path + path = [tokenIn, tokenOut]; + } else if ( + tokenIn != blockchain$1.wrapped.address && + tokenIn != blockchain$1.currency.address && + await pathExists$4({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$1.wrapped.address && + tokenOut != blockchain$1.currency.address && + await pathExists$4({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + ) { + // path via blockchain.wrapped.address + path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + } else if ( + !blockchain$1.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$1.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) + ) { + // path via TOKEN_IN <> STABLE <> TOKEN_OUT + path = [tokenIn, stable, tokenOut]; + } + + // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address + // because that actually reflects how things are routed in reality: + if(_optionalChain$3([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { + path.splice(1, 0, blockchain$1.wrapped.address); + } else if(_optionalChain$3([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { + path.splice(path.length-1, 0, blockchain$1.wrapped.address); + } + return { path, exchangePath: getExchangePath$3({ path }) } +}; + +let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + + let amounts = [ethers.BigNumber.from(amountIn || amountInMax)]; + + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts +}; + +let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + + path = path.slice().reverse(); + let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)]; + + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts.slice().reverse() +}; + +let getAmounts$4 = async ({ + path, + tokenIn, + tokenOut, + amountOut, + amountIn, + amountInMax, + amountOutMin +}) => { + path = getExchangePath$3({ path }); + let amounts; + if (amountOut) { + amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if (amountIn) { + amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } else if(amountOutMin) { + amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if(amountInMax) { + amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } + return { + amountOut: (amountOut || amountOutMin), + amountIn: (amountIn || amountInMax), + amountInMax: (amountInMax || amountIn), + amountOutMin: (amountOutMin || amountOut), + amounts + } +}; + +const blockchain = Blockchains.solana; + +const getTransaction$4 = async({ + path, + amountIn, + amountInMax, + amountOut, + amountOutMin, + amounts, + amountInInput, + amountOutInput, + amountInMaxInput, + amountOutMinInput, + account +})=>{ + let transaction = { blockchain: 'solana' }; + let instructions = []; + + const exchangePath = getExchangePath$3({ path }); + if(exchangePath.length > 3) { throw 'Raydium can only handle fixed paths with a max length of 3 (2 pools)!' } + const tokenIn = exchangePath[0]; + const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; + const tokenOut = exchangePath[exchangePath.length-1]; + + let pairs; + if(exchangePath.length == 2) { + pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + } else { + if(amountInInput || amountInMaxInput) { + pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + } else { // originally amountOut + pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + } + } + + let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let wrappedAccount; + const provider = await getProvider('solana'); + + if(startsWrapped || endsUnwrapped) { + const rent = await provider.getMinimumBalanceForRentExemption(Token.solana.TOKEN_LAYOUT.span); + const keypair = Keypair.generate(); + wrappedAccount = keypair.publicKey.toString(); + const lamports = startsWrapped ? new BN(amountIn.toString()).add(new BN(rent)) : new BN(rent); + let createAccountInstruction = SystemProgram.createAccount({ + fromPubkey: new PublicKey(account), + newAccountPubkey: new PublicKey(wrappedAccount), + programId: new PublicKey(Token.solana.TOKEN_PROGRAM), + space: Token.solana.TOKEN_LAYOUT.span, + lamports + }); + createAccountInstruction.signers = [keypair]; + instructions.push(createAccountInstruction); + instructions.push( + Token.solana.initializeAccountInstruction({ + account: wrappedAccount, + token: blockchain.wrapped.address, + owner: account + }) + ); + } + + if(pairs.length === 1) { // single hop + const tokenAccountIn = startsWrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenIn })); + const tokenAccountOut = endsUnwrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenOut })); + const pool = pairs[0]; + await request(`solana://${pool.data.marketId}/getAccountInfo`, { api: MARKET_LAYOUT }); + + let LAYOUT, data; + LAYOUT = struct([ + u8('instruction'), + u64$1('amountIn'), + u64$1('minAmountOut') + ]); + data = Buffer.alloc(LAYOUT.span); + LAYOUT.encode( + { + instruction: 9, + amountIn: new BN((amountIn || amountInMax).toString()), + minAmountOut: new BN((amountOutMin || amountOut).toString()), + }, + data, + ); + + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + + // let keys = [ + // // token_program + // { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // // amm + // { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // // amm authority + // { pubkey: await getAssociatedAuthority(new PublicKey(pool.publicKey)), isWritable: false, isSigner: false }, + // // amm openOrders + // { pubkey: new PublicKey(pool.data.openOrders), isWritable: true, isSigner: false }, + // // amm baseVault + // { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // // amm quoteVault + // { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // // openbook marketProgramId + // { pubkey: new PublicKey(pool.data.marketProgramId), isWritable: false, isSigner: false }, + // // openbook marketId + // { pubkey: new PublicKey(pool.data.marketId), isWritable: true, isSigner: false }, + // // openbook marketBids + // { pubkey: market.bids, isWritable: true, isSigner: false }, + // // openbook marketAsks + // { pubkey: market.asks, isWritable: true, isSigner: false }, + // // openbook eventQueue + // { pubkey: market.eventQueue, isWritable: true, isSigner: false }, + // // openbook baseVault + // { pubkey: market.baseVault, isWritable: true, isSigner: false }, + // // openbook quoteVault + // { pubkey: market.quoteVault, isWritable: true, isSigner: false }, + // // openbook marketAuthority + // { pubkey: await getAssociatedMarketAuthority(pool.data.marketProgramId, pool.data.marketId), isWritable: false, isSigner: false }, + // // user tokenAccountIn + // { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // // user tokenAccountOut + // { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // // user owner + // { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + // ] + + let keys = [ + // token_program + { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // amm + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm authority + { pubkey: new PublicKey('5Q544fKrFoe6tsEbD7S8EmxGTJYAKtTVhAW5Q5pge4j1'), isWritable: false, isSigner: false }, + // amm openOrders + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm baseVault + { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // amm quoteVault + { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // openbook marketProgramId + { pubkey: new PublicKey(pool.publicKey), isWritable: false, isSigner: false }, + // openbook marketId + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketBids + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAsks + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook eventQueue + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook baseVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook quoteVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAuthority + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // user tokenAccountIn + { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // user tokenAccountOut + { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // user owner + { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + ]; + + console.log('keys', JSON.stringify(keys)); + + instructions.push( + new TransactionInstruction({ + programId: new PublicKey('675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8'), + keys, + data, + }) + ); + } else if (pairs.length === 2) ; + + if(startsWrapped || endsUnwrapped) { + instructions.push( + Token.solana.closeAccountInstruction({ + account: wrappedAccount, + owner: account + }) + ); + } + + // await debug(instructions, provider) + + transaction.instructions = instructions; + return transaction +}; + +var Raydium = { + findPath: findPath$4, + pathExists: pathExists$4, + getAmounts: getAmounts$4, + getTransaction: getTransaction$4, + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, +}; + +// // AMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_amm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_amm.api.span }, +// { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 +// { memcmp: { offset: 400, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 432, bytes: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_amm.api, +// }) +// console.log(accounts) + + +// // CPAMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_cpamm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_cpamm.api.span }, +// { memcmp: { offset: 168, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 200, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_cpamm.api, +// }) +// console.log(accounts) + + +// // CLMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_clmm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_clmm.api.span }, +// { memcmp: { offset: 73, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 105, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_clmm.api, +// }) +// console.log(accounts) + +const exchange$g = { + + name: 'raydium', + label: 'Raydium', + logo: '', + protocol: 'raydium', + + slippage: true, + + blockchains: ['solana'], + + solana: { + + router_amm: { + address: '675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8', + api: Raydium.AMM_LAYOUT, + }, + + router_cpamm: { + address: 'CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C', + api: Raydium.CPAMM_LAYOUT + }, + + router_clmm: { + address: 'CAMMCzo5YL8w4VFF8KVHrK22GGUsp5VTaW7grrKgrWqK', + api: Raydium.CLMM_LAYOUT + }, + } +}; + +var raydium = (scope)=>{ + return new Exchange( Object.assign(exchange$g, { scope, - findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$g }), - pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$g }), - getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$g }), + findPath: (args)=>Raydium.findPath({ ...args, exchange: exchange$g }), + pathExists: (args)=>Raydium.pathExists({ ...args, exchange: exchange$g }), + getAmounts: (args)=>Raydium.getAmounts({ ...args, exchange: exchange$g }), getPrep: (args)=>{}, - getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$g }), + getTransaction: (args)=>Raydium.getTransaction({ ...args, exchange: exchange$g }), }) ) }; @@ -4812,6 +5621,7 @@ var wxdai = (scope)=>{ const exchanges = [ orca(), + raydium(), uniswap_v3(), pancakeswap_v3(), uniswap_v2(), @@ -4858,6 +5668,7 @@ exchanges.polygon.forEach((exchange)=>{ exchanges.polygon[exchange.name] = excha exchanges.solana = [ orca('solana'), + raydium('solana'), ]; exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange; }); diff --git a/dist/esm/index.solana.js b/dist/esm/index.solana.js index 30c1578..b90610d 100644 --- a/dist/esm/index.solana.js +++ b/dist/esm/index.solana.js @@ -2,10 +2,10 @@ import Token from '@depay/web3-tokens-solana'; import { request, getProvider } from '@depay/web3-client-solana'; import { ethers } from 'ethers'; import Blockchains from '@depay/web3-blockchains'; -import { BN, struct, publicKey, u128, u64 as u64$1, seq, u8, u16, i32, bool, i128, PublicKey, Buffer, Keypair, SystemProgram, TransactionInstruction } from '@depay/solana-web3.js'; +import { BN, struct, publicKey, u128, u64 as u64$1, seq, u8, u16, i32, bool, i128, PublicKey, Buffer, Keypair, SystemProgram, TransactionInstruction, blob, u32 } from '@depay/solana-web3.js'; import Decimal from 'decimal.js'; -function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { +function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { constructor({ blockchain, tokenIn, @@ -31,10 +31,10 @@ function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[ this.decimalsOut = decimalsOut; this.path = path; this.pools = pools; - this.amountIn = _optionalChain$1([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); - this.amountOutMin = _optionalChain$1([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); - this.amountOut = _optionalChain$1([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); - this.amountInMax = _optionalChain$1([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); + this.amountIn = _optionalChain$3([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); + this.amountOutMin = _optionalChain$3([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); + this.amountOut = _optionalChain$3([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); + this.amountInMax = _optionalChain$3([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); this.exchange = exchange; this.getPrep = getPrep; this.getTransaction = getTransaction; @@ -308,7 +308,10 @@ const route$1 = ({ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { @@ -1587,7 +1590,7 @@ let getAccounts = async (base, quote) => { return accounts }; -let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { +let getPairsWithPrice$4 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { try { let accounts = await getAccounts(tokenIn, tokenOut); if(accounts.length === 0) { accounts = await getAccounts(tokenOut, tokenIn); } @@ -1614,32 +1617,31 @@ let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amount } }; -let getHighestPrice = (pairs)=>{ +let getHighestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).gt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; -let getLowestPrice = (pairs)=>{ +let getLowestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).lt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; -let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { - const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); +let getBestPair$1 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice$4({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); if(!pairs || pairs.length === 0) { return } let bestPair; if(amountIn || amountInMax) { - bestPair = getHighestPrice(pairs); + bestPair = getHighestPrice$1(pairs); } else { // amount out - bestPair = getLowestPrice(pairs); + bestPair = getLowestPrice$1(pairs); } - return bestPair }; -function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } -const blockchain$1 = Blockchains.solana; +function _optionalChain$2(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } +const blockchain$3 = Blockchains.solana; // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. // @@ -1647,64 +1649,64 @@ const blockchain$1 = Blockchains.solana; // to be able to differentiate between SOL<>Token and WSOL<>Token swaps // as they are not the same! // -let getExchangePath = ({ path }) => { +let getExchangePath$1 = ({ path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( - token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && - path[index-1] != blockchain$1.wrapped.address + token === blockchain$3.currency.address && path[index+1] != blockchain$3.wrapped.address && + path[index-1] != blockchain$3.wrapped.address ) { - return blockchain$1.wrapped.address + return blockchain$3.wrapped.address } else { return token } }); - if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + if(exchangePath[0] == blockchain$3.currency.address && exchangePath[1] == blockchain$3.wrapped.address) { exchangePath.splice(0, 1); - } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + } else if(exchangePath[exchangePath.length-1] == blockchain$3.currency.address && exchangePath[exchangePath.length-2] == blockchain$3.wrapped.address) { exchangePath.splice(exchangePath.length-1, 1); } return exchangePath }; -let pathExists = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { +let pathExists$1 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { if(path.length == 1) { return false } - path = getExchangePath({ path }); - if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + path = getExchangePath$1({ path }); + if((await getPairsWithPrice$4({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { return true } else { return false } }; -let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { +let findPath$1 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { if( - [tokenIn, tokenOut].includes(blockchain$1.currency.address) && - [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + [tokenIn, tokenOut].includes(blockchain$3.currency.address) && + [tokenIn, tokenOut].includes(blockchain$3.wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path, stablesIn, stablesOut, stable; - if (await pathExists({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + if (await pathExists$1({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { // direct path path = [tokenIn, tokenOut]; } else if ( - tokenIn != blockchain$1.wrapped.address && - tokenIn != blockchain$1.currency.address && - await pathExists({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && - tokenOut != blockchain$1.wrapped.address && - tokenOut != blockchain$1.currency.address && - await pathExists({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + tokenIn != blockchain$3.wrapped.address && + tokenIn != blockchain$3.currency.address && + await pathExists$1({ path: [tokenIn, blockchain$3.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$3.wrapped.address && + tokenOut != blockchain$3.currency.address && + await pathExists$1({ path: [tokenOut, blockchain$3.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ) { // path via blockchain.wrapped.address - path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + path = [tokenIn, blockchain$3.wrapped.address, tokenOut]; } else if ( - !blockchain$1.stables.usd.includes(tokenIn) && - (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && - !blockchain$1.stables.usd.includes(tokenOut) && - (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$1({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$1({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) ) { // path via TOKEN_IN <> STABLE <> TOKEN_OUT @@ -1713,22 +1715,26 @@ let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amo // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address // because that actually reflects how things are routed in reality: - if(_optionalChain([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { - path.splice(1, 0, blockchain$1.wrapped.address); - } else if(_optionalChain([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { - path.splice(path.length-1, 0, blockchain$1.wrapped.address); + if(_optionalChain$2([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$3.currency.address) { + path.splice(1, 0, blockchain$3.wrapped.address); + } else if(_optionalChain$2([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$3.currency.address) { + path.splice(path.length-1, 0, blockchain$3.wrapped.address); } - return { path, exchangePath: getExchangePath({ path }) } + return { path, exchangePath: getExchangePath$1({ path }) } }; -let getAmountsOut = async ({ path, amountIn, amountInMax }) => { +let getAmountsOut$1 = async ({ path, amountIn, amountInMax }) => { let amounts = [ethers.BigNumber.from(amountIn || amountInMax)]; - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -1736,15 +1742,19 @@ let getAmountsOut = async ({ path, amountIn, amountInMax }) => { return amounts }; -let getAmountsIn = async({ path, amountOut, amountOutMin }) => { +let getAmountsIn$1 = async({ path, amountOut, amountOutMin }) => { path = path.slice().reverse(); let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)]; - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -1752,7 +1762,7 @@ let getAmountsIn = async({ path, amountOut, amountOutMin }) => { return amounts.slice().reverse() }; -let getAmounts = async ({ +let getAmounts$1 = async ({ path, tokenIn, tokenOut, @@ -1761,10 +1771,10 @@ let getAmounts = async ({ amountInMax, amountOutMin }) => { - path = getExchangePath({ path }); + path = getExchangePath$1({ path }); let amounts; if (amountOut) { - amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOut, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -1772,7 +1782,7 @@ let getAmounts = async ({ amountInMax = amountIn; } } else if (amountIn) { - amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountIn, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -1780,7 +1790,7 @@ let getAmounts = async ({ amountOutMin = amountOut; } } else if(amountOutMin) { - amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOutMin, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -1788,7 +1798,7 @@ let getAmounts = async ({ amountInMax = amountIn; } } else if(amountInMax) { - amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountInMax, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -1805,7 +1815,7 @@ let getAmounts = async ({ } }; -const blockchain = Blockchains.solana; +const blockchain$2 = Blockchains.solana; const SWAP_INSTRUCTION = new BN("14449647541112719096"); const TWO_HOP_SWAP_INSTRUCTION = new BN("16635068063392030915"); @@ -2020,7 +2030,7 @@ const getSwapInstructionData = ({ amount, otherAmountThreshold, sqrtPriceLimit, return data }; -const getTransaction = async ({ +const getTransaction$1 = async ({ path, amountIn, amountInMax, @@ -2036,7 +2046,7 @@ const getTransaction = async ({ let transaction = { blockchain: 'solana' }; let instructions = []; - const exchangePath = getExchangePath({ path }); + const exchangePath = getExchangePath$1({ path }); if(exchangePath.length > 3) { throw 'Orca can only handle fixed paths with a max length of 3 (2 pools)!' } const tokenIn = exchangePath[0]; const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; @@ -2044,19 +2054,19 @@ const getTransaction = async ({ let pairs; if(exchangePath.length == 2) { - pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + pairs = [await getBestPair$1({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; } else { if(amountInInput || amountInMaxInput) { - pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; - pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); } else { // originally amountOut - pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; - pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); } } - let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); - let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let startsWrapped = (path[0] === blockchain$2.currency.address && exchangePath[0] === blockchain$2.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain$2.currency.address && exchangePath[exchangePath.length-1] === blockchain$2.wrapped.address); let wrappedAccount; const provider = await getProvider('solana'); @@ -2077,7 +2087,7 @@ const getTransaction = async ({ instructions.push( Token.solana.initializeAccountInstruction({ account: wrappedAccount, - token: blockchain.wrapped.address, + token: blockchain$2.wrapped.address, owner: account }) ); @@ -2174,14 +2184,14 @@ const getTransaction = async ({ }; var Orca = { - findPath, - pathExists, - getAmounts, - getTransaction, + findPath: findPath$1, + pathExists: pathExists$1, + getAmounts: getAmounts$1, + getTransaction: getTransaction$1, WHIRLPOOL_LAYOUT, }; -const exchange = { +const exchange$1 = { name: 'orca', label: 'Orca', @@ -2202,22 +2212,822 @@ const exchange = { var orca = (scope)=>{ + return new Exchange( + + Object.assign(exchange$1, { + scope, + + findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$1 }), + pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$1 }), + getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$1 }), + getPrep: (args)=>{}, + getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$1 }), + }) + ) +}; + +// OpenBook Market +const MARKET_LAYOUT = struct([ + blob(5), + blob(8), // accountFlagsLayout('accountFlags'), + publicKey('ownAddress'), + u64$1('vaultSignerNonce'), + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('baseVault'), + u64$1('baseDepositsTotal'), + u64$1('baseFeesAccrued'), + publicKey('quoteVault'), + u64$1('quoteDepositsTotal'), + u64$1('quoteFeesAccrued'), + u64$1('quoteDustThreshold'), + publicKey('requestQueue'), + publicKey('eventQueue'), + publicKey('bids'), + publicKey('asks'), + u64$1('baseLotSize'), + u64$1('quoteLotSize'), + u64$1('feeRateBps'), + u64$1('referrerRebatesAccrued'), + blob(7), +]); + +// AMM 675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8 +const AMM_LAYOUT = struct([ + u64$1('status'), + u64$1('nonce'), + u64$1('maxOrder'), + u64$1('depth'), + u64$1('baseDecimal'), + u64$1('quoteDecimal'), + u64$1('state'), + u64$1('resetFlag'), + u64$1('minSize'), + u64$1('volMaxCutRatio'), + u64$1('amountWaveRatio'), + u64$1('baseLotSize'), + u64$1('quoteLotSize'), + u64$1('minPriceMultiplier'), + u64$1('maxPriceMultiplier'), + u64$1('systemDecimalValue'), + u64$1('minSeparateNumerator'), + u64$1('minSeparateDenominator'), + u64$1('tradeFeeNumerator'), + u64$1('tradeFeeDenominator'), + u64$1('pnlNumerator'), + u64$1('pnlDenominator'), + u64$1('swapFeeNumerator'), + u64$1('swapFeeDenominator'), + u64$1('baseNeedTakePnl'), + u64$1('quoteNeedTakePnl'), + u64$1('quoteTotalPnl'), + u64$1('baseTotalPnl'), + u64$1('poolOpenTime'), + u64$1('punishPcAmount'), + u64$1('punishCoinAmount'), + u64$1('orderbookToInitTime'), + u128('swapBaseInAmount'), + u128('swapQuoteOutAmount'), + u64$1('swapBase2QuoteFee'), + u128('swapQuoteInAmount'), + u128('swapBaseOutAmount'), + u64$1('swapQuote2BaseFee'), + // amm vault + publicKey('baseVault'), + publicKey('quoteVault'), + // mint + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('lpMint'), + // market + publicKey('openOrders'), + publicKey('marketId'), + publicKey('marketProgramId'), + publicKey('targetOrders'), + publicKey('withdrawQueue'), + publicKey('lpVault'), + publicKey('owner'), + // true circulating supply without lock up + u64$1('lpReserve'), + seq(u64$1(), 3, 'padding'), +]); + +// CP_AMM CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C +const CPAMM_LAYOUT = struct([ + blob(8), + publicKey("configId"), + publicKey("poolCreator"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("mintLp"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("mintProgramA"), + publicKey("mintProgramB"), + publicKey("observationId"), + u8("bump"), + u8("status"), + u8("lpDecimals"), + u8("mintDecimalA"), + u8("mintDecimalB"), + u64$1("lpAmount"), + u64$1("protocolFeesMintA"), + u64$1("protocolFeesMintB"), + u64$1("fundFeesMintA"), + u64$1("fundFeesMintB"), + u64$1("openTime"), + seq(u64$1(), 32), +]); + +// CLMM + +const RewardInfo = struct([ + u8("rewardState"), + u64$1("openTime"), + u64$1("endTime"), + u64$1("lastUpdateTime"), + u128("emissionsPerSecondX64"), + u64$1("rewardTotalEmissioned"), + u64$1("rewardClaimed"), + publicKey("tokenMint"), + publicKey("tokenVault"), + publicKey("creator"), + u128("rewardGrowthGlobalX64"), +]); + +const CLMM_LAYOUT = struct([ + blob(8), + u8("bump"), + publicKey("ammConfig"), + publicKey("creator"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("observationId"), + u8("mintDecimalsA"), + u8("mintDecimalsB"), + u16("tickSpacing"), + u128("liquidity"), + u128("sqrtPriceX64"), + i32("tickCurrent"), + u32(), + u128("feeGrowthGlobalX64A"), + u128("feeGrowthGlobalX64B"), + u64$1("protocolFeesTokenA"), + u64$1("protocolFeesTokenB"), + u128("swapInAmountTokenA"), + u128("swapOutAmountTokenB"), + u128("swapInAmountTokenB"), + u128("swapOutAmountTokenA"), + u8("status"), + seq(u8(), 7, ""), + seq(RewardInfo, 3, "rewardInfos"), + seq(u64$1(), 16, "tickArrayBitmap"), + u64$1("totalFeesTokenA"), + u64$1("totalFeesClaimedTokenA"), + u64$1("totalFeesTokenB"), + u64$1("totalFeesClaimedTokenB"), + u64$1("fundFeesTokenA"), + u64$1("fundFeesTokenB"), + u64$1("startTime"), + seq(u64$1(), 15 * 4 - 3, "padding"), +]); + +function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + +const LIQUIDITY_FEES_NUMERATOR = new BN(25); +const LIQUIDITY_FEES_DENOMINATOR = new BN(10000); + +const BNDivCeil = (bn1, bn2)=> { + const { div, mod } = bn1.divmod(bn2); + + if (mod.gt(new BN(0))) { + return div.add(new BN(1)) + } else { + return div + } +}; + +const getPairs = (base, quote)=>{ + return request(`solana://675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8/getProgramAccounts`, { + params: { filters: [ + { dataSize: AMM_LAYOUT.span }, + { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 (Swap) + { memcmp: { offset: 400, bytes: base }}, + { memcmp: { offset: 432, bytes: quote }}, + ]}, + api: AMM_LAYOUT, + cache: 86400, // 24h, + cacheKey: ['raydium', base.toString(), quote.toString()].join('-') + }) +}; + +const getPairsWithPrice$3 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + let accounts = await getPairs(tokenIn, tokenOut); + + if(accounts.length == 0) { + accounts = await getPairs(tokenOut, tokenIn); + } + + const pairs = await Promise.all(accounts.map(async(account)=>{ + + const baseMint = account.data.baseMint.toString(); + const quoteMint = account.data.quoteMint.toString(); + + const balances = await Promise.all([ + request(`solana://${account.data.baseVault.toString()}/getTokenAccountBalance`, { cache: 5 }), + request(`solana://${account.data.quoteVault.toString()}/getTokenAccountBalance`, { cache: 5 }) + ]); + const baseReserve = (_optionalChain$1([balances, 'access', _ => _[0], 'optionalAccess', _2 => _2.value, 'optionalAccess', _3 => _3.amount]) ? new BN(_optionalChain$1([balances, 'access', _4 => _4[0], 'optionalAccess', _5 => _5.value, 'optionalAccess', _6 => _6.amount])) : new BN(0)).sub(account.data.baseNeedTakePnl); + const quoteReserve = (_optionalChain$1([balances, 'access', _7 => _7[1], 'optionalAccess', _8 => _8.value, 'optionalAccess', _9 => _9.amount]) ? new BN(_optionalChain$1([balances, 'access', _10 => _10[1], 'optionalAccess', _11 => _11.value, 'optionalAccess', _12 => _12.amount])) : new BN(0)).sub(account.data.quoteNeedTakePnl); + + if(baseMint === Blockchains.solana.wrapped.address) { + if(baseReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (quoteMint === Blockchains.solana.wrapped.address) { + if(quoteReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(baseMint)) { + if((parseFloat(baseReserve.toString()) / 10 ** account.data.baseDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(quoteMint)) { + if((parseFloat(quoteReserve.toString()) / 10 ** account.data.quoteDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } + + const reserves = [baseReserve, quoteReserve]; + + if(tokenOut === baseMint) { + reserves.reverse(); + } + + const [reserveIn, reserveOut] = reserves; + + let price; + if(amountOut || amountOutMin) { // compute amountIn + + new BN(0); + let amountOutRaw = new BN((amountOut || amountOutMin).toString()); + + if (amountOutRaw.gt(reserveOut)) { + amountOutRaw = reserveOut.sub(new BN(1)); + } + + const denominator = reserveOut.sub(amountOutRaw); + const amountInWithoutFee = reserveIn.mul(amountOutRaw).div(denominator); + + price = amountInWithoutFee + .mul(LIQUIDITY_FEES_DENOMINATOR) + .div(LIQUIDITY_FEES_DENOMINATOR.sub(LIQUIDITY_FEES_NUMERATOR)) + .toString(); + + } else { // compute amountOut + + new BN(0); + const amountInRaw = new BN((amountIn || amountInMin).toString()); + const feeRaw = BNDivCeil(amountInRaw.mul(LIQUIDITY_FEES_NUMERATOR), LIQUIDITY_FEES_DENOMINATOR); + + const amountInWithFee = amountInRaw.sub(feeRaw); + const denominator = reserveIn.add(amountInWithFee); + + price = reserveOut.mul(amountInWithFee).div(denominator).toString(); + } + + return { + publicKey: account.pubkey.toString(), + baseMint, + quoteMint, + price, + data: account.data + } + + })); + + return pairs.filter(Boolean) +}; + +const getPairsWithPrice$2 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +}; + +const getPairsWithPrice$1 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +}; + +const getParisWithPriceForAllTypes = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + return Promise.all([ + getPairsWithPrice$3({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$2({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$1({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + ]).then((pairsAMM, pairsCPAMM, pairsCLMNN)=>{ + return [ + (pairsAMM || []).filter(Boolean).flat(), + (pairsCPAMM || []).filter(Boolean).flat(), + (pairsCLMNN || []).filter(Boolean).flat() + ].flat() + }) +}; + +const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + try { + return await getParisWithPriceForAllTypes({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + } catch (e) { + return [] + } +}; + +let getHighestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).gt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +}; + +let getLowestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).lt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +}; + +let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + + if(!pairs || pairs.length === 0) { return } + + let bestPair; + + if(amountIn || amountInMax) { + bestPair = getHighestPrice(pairs); + } else { // amount out + bestPair = getLowestPrice(pairs); + } + + return bestPair +}; + +function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } +const blockchain$1 = Blockchains.solana; + +// Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. +// +// We keep 11111111111111111111111111111111 internally +// to be able to differentiate between SOL<>Token and WSOL<>Token swaps +// as they are not the same! +// +let getExchangePath = ({ path }) => { + if(!path) { return } + let exchangePath = path.map((token, index) => { + if ( + token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && + path[index-1] != blockchain$1.wrapped.address + ) { + return blockchain$1.wrapped.address + } else { + return token + } + }); + + if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + exchangePath.splice(0, 1); + } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + exchangePath.splice(exchangePath.length-1, 1); + } + + return exchangePath +}; + +let pathExists = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + if(path.length == 1) { return false } + path = getExchangePath({ path }); + if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + return true + } else { + return false + } +}; + +let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + if( + [tokenIn, tokenOut].includes(blockchain$1.currency.address) && + [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + ) { return { path: undefined, exchangePath: undefined } } + + let path, stablesIn, stablesOut, stable; + + if (await pathExists({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + // direct path + path = [tokenIn, tokenOut]; + } else if ( + tokenIn != blockchain$1.wrapped.address && + tokenIn != blockchain$1.currency.address && + await pathExists({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$1.wrapped.address && + tokenOut != blockchain$1.currency.address && + await pathExists({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + ) { + // path via blockchain.wrapped.address + path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + } else if ( + !blockchain$1.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$1.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) + ) { + // path via TOKEN_IN <> STABLE <> TOKEN_OUT + path = [tokenIn, stable, tokenOut]; + } + + // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address + // because that actually reflects how things are routed in reality: + if(_optionalChain([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { + path.splice(1, 0, blockchain$1.wrapped.address); + } else if(_optionalChain([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { + path.splice(path.length-1, 0, blockchain$1.wrapped.address); + } + return { path, exchangePath: getExchangePath({ path }) } +}; + +let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + + let amounts = [ethers.BigNumber.from(amountIn || amountInMax)]; + + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts +}; + +let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + + path = path.slice().reverse(); + let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)]; + + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts.slice().reverse() +}; + +let getAmounts = async ({ + path, + tokenIn, + tokenOut, + amountOut, + amountIn, + amountInMax, + amountOutMin +}) => { + path = getExchangePath({ path }); + let amounts; + if (amountOut) { + amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if (amountIn) { + amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } else if(amountOutMin) { + amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if(amountInMax) { + amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } + return { + amountOut: (amountOut || amountOutMin), + amountIn: (amountIn || amountInMax), + amountInMax: (amountInMax || amountIn), + amountOutMin: (amountOutMin || amountOut), + amounts + } +}; + +const blockchain = Blockchains.solana; + +const getTransaction = async({ + path, + amountIn, + amountInMax, + amountOut, + amountOutMin, + amounts, + amountInInput, + amountOutInput, + amountInMaxInput, + amountOutMinInput, + account +})=>{ + let transaction = { blockchain: 'solana' }; + let instructions = []; + + const exchangePath = getExchangePath({ path }); + if(exchangePath.length > 3) { throw 'Raydium can only handle fixed paths with a max length of 3 (2 pools)!' } + const tokenIn = exchangePath[0]; + const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; + const tokenOut = exchangePath[exchangePath.length-1]; + + let pairs; + if(exchangePath.length == 2) { + pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + } else { + if(amountInInput || amountInMaxInput) { + pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + } else { // originally amountOut + pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + } + } + + let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let wrappedAccount; + const provider = await getProvider('solana'); + + if(startsWrapped || endsUnwrapped) { + const rent = await provider.getMinimumBalanceForRentExemption(Token.solana.TOKEN_LAYOUT.span); + const keypair = Keypair.generate(); + wrappedAccount = keypair.publicKey.toString(); + const lamports = startsWrapped ? new BN(amountIn.toString()).add(new BN(rent)) : new BN(rent); + let createAccountInstruction = SystemProgram.createAccount({ + fromPubkey: new PublicKey(account), + newAccountPubkey: new PublicKey(wrappedAccount), + programId: new PublicKey(Token.solana.TOKEN_PROGRAM), + space: Token.solana.TOKEN_LAYOUT.span, + lamports + }); + createAccountInstruction.signers = [keypair]; + instructions.push(createAccountInstruction); + instructions.push( + Token.solana.initializeAccountInstruction({ + account: wrappedAccount, + token: blockchain.wrapped.address, + owner: account + }) + ); + } + + if(pairs.length === 1) { // single hop + const tokenAccountIn = startsWrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenIn })); + const tokenAccountOut = endsUnwrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenOut })); + const pool = pairs[0]; + await request(`solana://${pool.data.marketId}/getAccountInfo`, { api: MARKET_LAYOUT }); + + let LAYOUT, data; + LAYOUT = struct([ + u8('instruction'), + u64$1('amountIn'), + u64$1('minAmountOut') + ]); + data = Buffer.alloc(LAYOUT.span); + LAYOUT.encode( + { + instruction: 9, + amountIn: new BN((amountIn || amountInMax).toString()), + minAmountOut: new BN((amountOutMin || amountOut).toString()), + }, + data, + ); + + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + + // let keys = [ + // // token_program + // { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // // amm + // { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // // amm authority + // { pubkey: await getAssociatedAuthority(new PublicKey(pool.publicKey)), isWritable: false, isSigner: false }, + // // amm openOrders + // { pubkey: new PublicKey(pool.data.openOrders), isWritable: true, isSigner: false }, + // // amm baseVault + // { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // // amm quoteVault + // { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // // openbook marketProgramId + // { pubkey: new PublicKey(pool.data.marketProgramId), isWritable: false, isSigner: false }, + // // openbook marketId + // { pubkey: new PublicKey(pool.data.marketId), isWritable: true, isSigner: false }, + // // openbook marketBids + // { pubkey: market.bids, isWritable: true, isSigner: false }, + // // openbook marketAsks + // { pubkey: market.asks, isWritable: true, isSigner: false }, + // // openbook eventQueue + // { pubkey: market.eventQueue, isWritable: true, isSigner: false }, + // // openbook baseVault + // { pubkey: market.baseVault, isWritable: true, isSigner: false }, + // // openbook quoteVault + // { pubkey: market.quoteVault, isWritable: true, isSigner: false }, + // // openbook marketAuthority + // { pubkey: await getAssociatedMarketAuthority(pool.data.marketProgramId, pool.data.marketId), isWritable: false, isSigner: false }, + // // user tokenAccountIn + // { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // // user tokenAccountOut + // { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // // user owner + // { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + // ] + + let keys = [ + // token_program + { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // amm + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm authority + { pubkey: new PublicKey('5Q544fKrFoe6tsEbD7S8EmxGTJYAKtTVhAW5Q5pge4j1'), isWritable: false, isSigner: false }, + // amm openOrders + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm baseVault + { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // amm quoteVault + { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // openbook marketProgramId + { pubkey: new PublicKey(pool.publicKey), isWritable: false, isSigner: false }, + // openbook marketId + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketBids + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAsks + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook eventQueue + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook baseVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook quoteVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAuthority + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // user tokenAccountIn + { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // user tokenAccountOut + { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // user owner + { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + ]; + + console.log('keys', JSON.stringify(keys)); + + instructions.push( + new TransactionInstruction({ + programId: new PublicKey('675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8'), + keys, + data, + }) + ); + } else if (pairs.length === 2) ; + + if(startsWrapped || endsUnwrapped) { + instructions.push( + Token.solana.closeAccountInstruction({ + account: wrappedAccount, + owner: account + }) + ); + } + + // await debug(instructions, provider) + + transaction.instructions = instructions; + return transaction +}; + +var Raydium = { + findPath, + pathExists, + getAmounts, + getTransaction, + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, +}; + +// // AMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_amm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_amm.api.span }, +// { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 +// { memcmp: { offset: 400, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 432, bytes: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_amm.api, +// }) +// console.log(accounts) + + +// // CPAMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_cpamm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_cpamm.api.span }, +// { memcmp: { offset: 168, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 200, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_cpamm.api, +// }) +// console.log(accounts) + + +// // CLMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_clmm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_clmm.api.span }, +// { memcmp: { offset: 73, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 105, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_clmm.api, +// }) +// console.log(accounts) + +const exchange = { + + name: 'raydium', + label: 'Raydium', + logo: '', + protocol: 'raydium', + + slippage: true, + + blockchains: ['solana'], + + solana: { + + router_amm: { + address: '675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8', + api: Raydium.AMM_LAYOUT, + }, + + router_cpamm: { + address: 'CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C', + api: Raydium.CPAMM_LAYOUT + }, + + router_clmm: { + address: 'CAMMCzo5YL8w4VFF8KVHrK22GGUsp5VTaW7grrKgrWqK', + api: Raydium.CLMM_LAYOUT + }, + } +}; + +var raydium = (scope)=>{ + return new Exchange( Object.assign(exchange, { scope, - findPath: (args)=>Orca.findPath({ ...args, exchange }), - pathExists: (args)=>Orca.pathExists({ ...args, exchange }), - getAmounts: (args)=>Orca.getAmounts({ ...args, exchange }), + findPath: (args)=>Raydium.findPath({ ...args, exchange }), + pathExists: (args)=>Raydium.pathExists({ ...args, exchange }), + getAmounts: (args)=>Raydium.getAmounts({ ...args, exchange }), getPrep: (args)=>{}, - getTransaction: (args)=>Orca.getTransaction({ ...args, exchange }), + getTransaction: (args)=>Raydium.getTransaction({ ...args, exchange }), }) ) }; const exchanges = [ orca(), + raydium(), ]; exchanges.forEach((exchange)=>{ exchanges[exchange.name] = exchange; @@ -2225,6 +3035,7 @@ exchanges.forEach((exchange)=>{ exchanges.solana = [ orca('solana'), + raydium('solana'), ]; exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange; }); diff --git a/dist/umd/index.evm.js b/dist/umd/index.evm.js index 21bfad6..d10210a 100644 --- a/dist/umd/index.evm.js +++ b/dist/umd/index.evm.js @@ -312,7 +312,10 @@ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { diff --git a/dist/umd/index.js b/dist/umd/index.js index 1c1a124..4689c42 100644 --- a/dist/umd/index.js +++ b/dist/umd/index.js @@ -10,7 +10,7 @@ var Blockchains__default = /*#__PURE__*/_interopDefaultLegacy(Blockchains); var Decimal__default = /*#__PURE__*/_interopDefaultLegacy(Decimal); - function _optionalChain$5(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { + function _optionalChain$7(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { constructor({ blockchain, tokenIn, @@ -36,10 +36,10 @@ this.decimalsOut = decimalsOut; this.path = path; this.pools = pools; - this.amountIn = _optionalChain$5([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); - this.amountOutMin = _optionalChain$5([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); - this.amountOut = _optionalChain$5([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); - this.amountInMax = _optionalChain$5([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); + this.amountIn = _optionalChain$7([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); + this.amountOutMin = _optionalChain$7([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); + this.amountOut = _optionalChain$7([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); + this.amountInMax = _optionalChain$7([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); this.exchange = exchange; this.getPrep = getPrep; this.getTransaction = getTransaction; @@ -313,7 +313,10 @@ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { @@ -437,7 +440,7 @@ } } - function _optionalChain$4(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + function _optionalChain$6(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } // Replaces 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE with the wrapped token and implies wrapping. // @@ -445,7 +448,7 @@ // to be able to differentiate between ETH<>Token and WETH<>Token swaps // as they are not the same! // - const getExchangePath$4 = ({ blockchain, exchange, path }) => { + const getExchangePath$5 = ({ blockchain, exchange, path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( @@ -477,8 +480,8 @@ } }; - const pathExists$5 = async ({ blockchain, exchange, path }) => { - const exchangePath = getExchangePath$4({ blockchain, exchange, path }); + const pathExists$6 = async ({ blockchain, exchange, path }) => { + const exchangePath = getExchangePath$5({ blockchain, exchange, path }); if(!exchangePath || exchangePath.length === 1) { return false } try { let pair = await web3Client.request({ @@ -487,7 +490,7 @@ method: 'getPair', api: exchange[blockchain].factory.api, cache: 3600000, - params: getExchangePath$4({ blockchain, exchange, path }), + params: getExchangePath$5({ blockchain, exchange, path }), }); if(!pair || pair == Blockchains__default['default'][blockchain].zero) { return false } let [reserves, token0, token1] = await Promise.all([ @@ -508,61 +511,61 @@ } catch (e){ console.log('e', e); return false } }; - const findPath$5 = async ({ blockchain, exchange, tokenIn, tokenOut }) => { + const findPath$6 = async ({ blockchain, exchange, tokenIn, tokenOut }) => { if( [tokenIn, tokenOut].includes(Blockchains__default['default'][blockchain].currency.address) && [tokenIn, tokenOut].includes(Blockchains__default['default'][blockchain].wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path; - if (await pathExists$5({ blockchain, exchange, path: [tokenIn, tokenOut] })) { + if (await pathExists$6({ blockchain, exchange, path: [tokenIn, tokenOut] })) { // direct path path = [tokenIn, tokenOut]; } else if ( tokenIn != Blockchains__default['default'][blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenIn, Blockchains__default['default'][blockchain].wrapped.address] }) && + await pathExists$6({ blockchain, exchange, path: [tokenIn, Blockchains__default['default'][blockchain].wrapped.address] }) && tokenOut != Blockchains__default['default'][blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenOut, Blockchains__default['default'][blockchain].wrapped.address] }) + await pathExists$6({ blockchain, exchange, path: [tokenOut, Blockchains__default['default'][blockchain].wrapped.address] }) ) { // path via WRAPPED path = [tokenIn, Blockchains__default['default'][blockchain].wrapped.address, tokenOut]; } else if ( !Blockchains__default['default'][blockchain].stables.usd.includes(tokenIn) && - (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [tokenIn, stable] })))).filter(Boolean).length && + (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [tokenIn, stable] })))).filter(Boolean).length && tokenOut != Blockchains__default['default'][blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [Blockchains__default['default'][blockchain].wrapped.address, tokenOut] }) + await pathExists$6({ blockchain, exchange, path: [Blockchains__default['default'][blockchain].wrapped.address, tokenOut] }) ) { // path via tokenIn -> USD -> WRAPPED -> tokenOut - let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [tokenIn, stable] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [tokenIn, stable] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, USD, Blockchains__default['default'][blockchain].wrapped.address, tokenOut]; } else if ( tokenIn != Blockchains__default['default'][blockchain].wrapped.address && - await pathExists$5({ blockchain, exchange, path: [tokenIn, Blockchains__default['default'][blockchain].wrapped.address] }) && + await pathExists$6({ blockchain, exchange, path: [tokenIn, Blockchains__default['default'][blockchain].wrapped.address] }) && !Blockchains__default['default'][blockchain].stables.usd.includes(tokenOut) && - (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length + (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length ) { // path via tokenIn -> WRAPPED -> USD -> tokenOut - let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, Blockchains__default['default'][blockchain].wrapped.address, USD, tokenOut]; } else if ( tokenIn != Blockchains__default['default'][blockchain].wrapped.address && tokenIn === Blockchains__default['default'][blockchain].currency.address && !Blockchains__default['default'][blockchain].stables.usd.includes(tokenOut) && - (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$5({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length + (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map((stable)=>pathExists$6({ blockchain, exchange, path: [stable, tokenOut] })))).filter(Boolean).length ) { - let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$5({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); + let USD = (await Promise.all(Blockchains__default['default'][blockchain].stables.usd.map(async (stable)=>{ return(await pathExists$6({ blockchain, exchange, path: [stable, tokenOut] }) ? stable : undefined) }))).find(Boolean); path = [tokenIn, USD, tokenOut]; } // Add WRAPPED to route path if things start or end with NATIVE // because that actually reflects how things are routed in reality: - if(_optionalChain$4([path, 'optionalAccess', _ => _.length]) && path[0] == Blockchains__default['default'][blockchain].currency.address) { + if(_optionalChain$6([path, 'optionalAccess', _ => _.length]) && path[0] == Blockchains__default['default'][blockchain].currency.address) { path.splice(1, 0, Blockchains__default['default'][blockchain].wrapped.address); - } else if(_optionalChain$4([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == Blockchains__default['default'][blockchain].currency.address) { + } else if(_optionalChain$6([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == Blockchains__default['default'][blockchain].currency.address) { path.splice(path.length-1, 0, Blockchains__default['default'][blockchain].wrapped.address); } - return { path, exchangePath: getExchangePath$4({ blockchain, exchange, path }) } + return { path, exchangePath: getExchangePath$5({ blockchain, exchange, path }) } }; let getAmountOut$3 = ({ blockchain, exchange, path, amountIn, tokenIn, tokenOut }) => { @@ -574,7 +577,7 @@ api: exchange[blockchain].router.api, params: { amountIn: amountIn, - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), }, }) .then((amountsOut)=>{ @@ -593,7 +596,7 @@ api: exchange[blockchain].router.api, params: { amountOut: amountOut, - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), }, block }) @@ -602,7 +605,7 @@ }) }; - let getAmounts$5 = async ({ + let getAmounts$6 = async ({ blockchain, exchange, path, @@ -679,7 +682,7 @@ }; - let getTransaction$5 = ({ + let getTransaction$6 = ({ exchange, blockchain, path, @@ -730,7 +733,7 @@ } transaction.params = Object.assign({}, transaction.params, { - path: getExchangePath$4({ blockchain, exchange, path }), + path: getExchangePath$5({ blockchain, exchange, path }), to: account, deadline: Math.round(Date.now() / 1000) + 60 * 60 * 24, // 24 hours }); @@ -743,17 +746,17 @@ const PAIR$1 = [{"inputs":[],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1","type":"uint256"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"Burn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount0In","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1In","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount0Out","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount1Out","type":"uint256"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint112","name":"reserve0","type":"uint112"},{"indexed":false,"internalType":"uint112","name":"reserve1","type":"uint112"}],"name":"Sync","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"constant":true,"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"MINIMUM_LIQUIDITY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"PERMIT_TYPEHASH","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"burn","outputs":[{"internalType":"uint256","name":"amount0","type":"uint256"},{"internalType":"uint256","name":"amount1","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"factory","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"getReserves","outputs":[{"internalType":"uint112","name":"_reserve0","type":"uint112"},{"internalType":"uint112","name":"_reserve1","type":"uint112"},{"internalType":"uint32","name":"_blockTimestampLast","type":"uint32"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"_token0","type":"address"},{"internalType":"address","name":"_token1","type":"address"}],"name":"initialize","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"kLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"mint","outputs":[{"internalType":"uint256","name":"liquidity","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"price0CumulativeLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"price1CumulativeLast","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"skim","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"uint256","name":"amount0Out","type":"uint256"},{"internalType":"uint256","name":"amount1Out","type":"uint256"},{"internalType":"address","name":"to","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"swap","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"sync","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"token0","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"token1","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]; var UniswapV2 = { - findPath: findPath$5, - pathExists: pathExists$5, - getAmounts: getAmounts$5, + findPath: findPath$6, + pathExists: pathExists$6, + getAmounts: getAmounts$6, getPrep: getPrep$3, - getTransaction: getTransaction$5, + getTransaction: getTransaction$6, ROUTER: ROUTER$3, FACTORY: FACTORY$3, PAIR: PAIR$1, }; - const exchange$h = { + const exchange$i = { name: 'honeyswap', label: 'Honeyswap', @@ -783,13 +786,13 @@ return new Exchange( - Object.assign(exchange$h, { + Object.assign(exchange$i, { scope, - findPath: (args)=>UniswapV2.findPath({ ...args, exchange: exchange$h }), - pathExists: (args)=>UniswapV2.pathExists({ ...args, exchange: exchange$h }), - getAmounts: (args)=>UniswapV2.getAmounts({ ...args, exchange: exchange$h }), - getPrep: (args)=>UniswapV2.getPrep({ ...args, exchange: exchange$h }), - getTransaction: (args)=>UniswapV2.getTransaction({ ...args, exchange: exchange$h }), + findPath: (args)=>UniswapV2.findPath({ ...args, exchange: exchange$i }), + pathExists: (args)=>UniswapV2.pathExists({ ...args, exchange: exchange$i }), + getAmounts: (args)=>UniswapV2.getAmounts({ ...args, exchange: exchange$i }), + getPrep: (args)=>UniswapV2.getPrep({ ...args, exchange: exchange$i }), + getTransaction: (args)=>UniswapV2.getTransaction({ ...args, exchange: exchange$i }), }) ) }; @@ -1949,7 +1952,7 @@ return accounts }; - let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + let getPairsWithPrice$4 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { try { let accounts = await getAccounts(tokenIn, tokenOut); if(accounts.length === 0) { accounts = await getAccounts(tokenOut, tokenIn); } @@ -1976,32 +1979,31 @@ } }; - let getHighestPrice = (pairs)=>{ + let getHighestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).gt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; - let getLowestPrice = (pairs)=>{ + let getLowestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).lt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; - let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { - const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + let getBestPair$1 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice$4({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); if(!pairs || pairs.length === 0) { return } let bestPair; if(amountIn || amountInMax) { - bestPair = getHighestPrice(pairs); + bestPair = getHighestPrice$1(pairs); } else { // amount out - bestPair = getLowestPrice(pairs); + bestPair = getLowestPrice$1(pairs); } - return bestPair }; - function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } - const blockchain$1 = Blockchains__default['default'].solana; + function _optionalChain$5(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + const blockchain$3 = Blockchains__default['default'].solana; // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. // @@ -2009,64 +2011,64 @@ // to be able to differentiate between SOL<>Token and WSOL<>Token swaps // as they are not the same! // - let getExchangePath$3 = ({ path }) => { + let getExchangePath$4 = ({ path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( - token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && - path[index-1] != blockchain$1.wrapped.address + token === blockchain$3.currency.address && path[index+1] != blockchain$3.wrapped.address && + path[index-1] != blockchain$3.wrapped.address ) { - return blockchain$1.wrapped.address + return blockchain$3.wrapped.address } else { return token } }); - if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + if(exchangePath[0] == blockchain$3.currency.address && exchangePath[1] == blockchain$3.wrapped.address) { exchangePath.splice(0, 1); - } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + } else if(exchangePath[exchangePath.length-1] == blockchain$3.currency.address && exchangePath[exchangePath.length-2] == blockchain$3.wrapped.address) { exchangePath.splice(exchangePath.length-1, 1); } return exchangePath }; - let pathExists$4 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + let pathExists$5 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { if(path.length == 1) { return false } - path = getExchangePath$3({ path }); - if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + path = getExchangePath$4({ path }); + if((await getPairsWithPrice$4({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { return true } else { return false } }; - let findPath$4 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + let findPath$5 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { if( - [tokenIn, tokenOut].includes(blockchain$1.currency.address) && - [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + [tokenIn, tokenOut].includes(blockchain$3.currency.address) && + [tokenIn, tokenOut].includes(blockchain$3.wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path, stablesIn, stablesOut, stable; - if (await pathExists$4({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + if (await pathExists$5({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { // direct path path = [tokenIn, tokenOut]; } else if ( - tokenIn != blockchain$1.wrapped.address && - tokenIn != blockchain$1.currency.address && - await pathExists$4({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && - tokenOut != blockchain$1.wrapped.address && - tokenOut != blockchain$1.currency.address && - await pathExists$4({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + tokenIn != blockchain$3.wrapped.address && + tokenIn != blockchain$3.currency.address && + await pathExists$5({ path: [tokenIn, blockchain$3.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$3.wrapped.address && + tokenOut != blockchain$3.currency.address && + await pathExists$5({ path: [tokenOut, blockchain$3.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ) { // path via blockchain.wrapped.address - path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + path = [tokenIn, blockchain$3.wrapped.address, tokenOut]; } else if ( - !blockchain$1.stables.usd.includes(tokenIn) && - (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && - !blockchain$1.stables.usd.includes(tokenOut) && - (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$5({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$5({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) ) { // path via TOKEN_IN <> STABLE <> TOKEN_OUT @@ -2075,22 +2077,26 @@ // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address // because that actually reflects how things are routed in reality: - if(_optionalChain$3([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { - path.splice(1, 0, blockchain$1.wrapped.address); - } else if(_optionalChain$3([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { - path.splice(path.length-1, 0, blockchain$1.wrapped.address); + if(_optionalChain$5([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$3.currency.address) { + path.splice(1, 0, blockchain$3.wrapped.address); + } else if(_optionalChain$5([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$3.currency.address) { + path.splice(path.length-1, 0, blockchain$3.wrapped.address); } - return { path, exchangePath: getExchangePath$3({ path }) } + return { path, exchangePath: getExchangePath$4({ path }) } }; - let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + let getAmountsOut$1 = async ({ path, amountIn, amountInMax }) => { let amounts = [ethers.ethers.BigNumber.from(amountIn || amountInMax)]; - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -2098,15 +2104,19 @@ return amounts }; - let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + let getAmountsIn$1 = async({ path, amountOut, amountOutMin }) => { path = path.slice().reverse(); let amounts = [ethers.ethers.BigNumber.from(amountOut || amountOutMin)]; - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -2114,7 +2124,7 @@ return amounts.slice().reverse() }; - let getAmounts$4 = async ({ + let getAmounts$5 = async ({ path, tokenIn, tokenOut, @@ -2123,10 +2133,10 @@ amountInMax, amountOutMin }) => { - path = getExchangePath$3({ path }); + path = getExchangePath$4({ path }); let amounts; if (amountOut) { - amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOut, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -2134,7 +2144,7 @@ amountInMax = amountIn; } } else if (amountIn) { - amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountIn, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -2142,7 +2152,7 @@ amountOutMin = amountOut; } } else if(amountOutMin) { - amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOutMin, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -2150,7 +2160,7 @@ amountInMax = amountIn; } } else if(amountInMax) { - amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountInMax, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -2167,7 +2177,7 @@ } }; - const blockchain = Blockchains__default['default'].solana; + const blockchain$2 = Blockchains__default['default'].solana; const SWAP_INSTRUCTION = new solanaWeb3_js.BN("14449647541112719096"); const TWO_HOP_SWAP_INSTRUCTION = new solanaWeb3_js.BN("16635068063392030915"); @@ -2382,7 +2392,7 @@ return data }; - const getTransaction$4 = async ({ + const getTransaction$5 = async ({ path, amountIn, amountInMax, @@ -2398,7 +2408,7 @@ let transaction = { blockchain: 'solana' }; let instructions = []; - const exchangePath = getExchangePath$3({ path }); + const exchangePath = getExchangePath$4({ path }); if(exchangePath.length > 3) { throw 'Orca can only handle fixed paths with a max length of 3 (2 pools)!' } const tokenIn = exchangePath[0]; const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; @@ -2406,19 +2416,19 @@ let pairs; if(exchangePath.length == 2) { - pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + pairs = [await getBestPair$1({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; } else { if(amountInInput || amountInMaxInput) { - pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; - pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); } else { // originally amountOut - pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; - pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); } } - let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); - let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let startsWrapped = (path[0] === blockchain$2.currency.address && exchangePath[0] === blockchain$2.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain$2.currency.address && exchangePath[exchangePath.length-1] === blockchain$2.wrapped.address); let wrappedAccount; const provider = await web3Client.getProvider('solana'); @@ -2439,7 +2449,7 @@ instructions.push( Token__default['default'].solana.initializeAccountInstruction({ account: wrappedAccount, - token: blockchain.wrapped.address, + token: blockchain$2.wrapped.address, owner: account }) ); @@ -2536,14 +2546,14 @@ }; var Orca = { - findPath: findPath$4, - pathExists: pathExists$4, - getAmounts: getAmounts$4, - getTransaction: getTransaction$4, + findPath: findPath$5, + pathExists: pathExists$5, + getAmounts: getAmounts$5, + getTransaction: getTransaction$5, WHIRLPOOL_LAYOUT, }; - const exchange$g = { + const exchange$h = { name: 'orca', label: 'Orca', @@ -2564,16 +2574,815 @@ var orca = (scope)=>{ + return new Exchange( + + Object.assign(exchange$h, { + scope, + + findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$h }), + pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$h }), + getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$h }), + getPrep: (args)=>{}, + getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$h }), + }) + ) + }; + + // OpenBook Market + const MARKET_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(5), + solanaWeb3_js.blob(8), // accountFlagsLayout('accountFlags'), + solanaWeb3_js.publicKey('ownAddress'), + solanaWeb3_js.u64('vaultSignerNonce'), + solanaWeb3_js.publicKey('baseMint'), + solanaWeb3_js.publicKey('quoteMint'), + solanaWeb3_js.publicKey('baseVault'), + solanaWeb3_js.u64('baseDepositsTotal'), + solanaWeb3_js.u64('baseFeesAccrued'), + solanaWeb3_js.publicKey('quoteVault'), + solanaWeb3_js.u64('quoteDepositsTotal'), + solanaWeb3_js.u64('quoteFeesAccrued'), + solanaWeb3_js.u64('quoteDustThreshold'), + solanaWeb3_js.publicKey('requestQueue'), + solanaWeb3_js.publicKey('eventQueue'), + solanaWeb3_js.publicKey('bids'), + solanaWeb3_js.publicKey('asks'), + solanaWeb3_js.u64('baseLotSize'), + solanaWeb3_js.u64('quoteLotSize'), + solanaWeb3_js.u64('feeRateBps'), + solanaWeb3_js.u64('referrerRebatesAccrued'), + solanaWeb3_js.blob(7), + ]); + + // AMM 675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8 + const AMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.u64('status'), + solanaWeb3_js.u64('nonce'), + solanaWeb3_js.u64('maxOrder'), + solanaWeb3_js.u64('depth'), + solanaWeb3_js.u64('baseDecimal'), + solanaWeb3_js.u64('quoteDecimal'), + solanaWeb3_js.u64('state'), + solanaWeb3_js.u64('resetFlag'), + solanaWeb3_js.u64('minSize'), + solanaWeb3_js.u64('volMaxCutRatio'), + solanaWeb3_js.u64('amountWaveRatio'), + solanaWeb3_js.u64('baseLotSize'), + solanaWeb3_js.u64('quoteLotSize'), + solanaWeb3_js.u64('minPriceMultiplier'), + solanaWeb3_js.u64('maxPriceMultiplier'), + solanaWeb3_js.u64('systemDecimalValue'), + solanaWeb3_js.u64('minSeparateNumerator'), + solanaWeb3_js.u64('minSeparateDenominator'), + solanaWeb3_js.u64('tradeFeeNumerator'), + solanaWeb3_js.u64('tradeFeeDenominator'), + solanaWeb3_js.u64('pnlNumerator'), + solanaWeb3_js.u64('pnlDenominator'), + solanaWeb3_js.u64('swapFeeNumerator'), + solanaWeb3_js.u64('swapFeeDenominator'), + solanaWeb3_js.u64('baseNeedTakePnl'), + solanaWeb3_js.u64('quoteNeedTakePnl'), + solanaWeb3_js.u64('quoteTotalPnl'), + solanaWeb3_js.u64('baseTotalPnl'), + solanaWeb3_js.u64('poolOpenTime'), + solanaWeb3_js.u64('punishPcAmount'), + solanaWeb3_js.u64('punishCoinAmount'), + solanaWeb3_js.u64('orderbookToInitTime'), + solanaWeb3_js.u128('swapBaseInAmount'), + solanaWeb3_js.u128('swapQuoteOutAmount'), + solanaWeb3_js.u64('swapBase2QuoteFee'), + solanaWeb3_js.u128('swapQuoteInAmount'), + solanaWeb3_js.u128('swapBaseOutAmount'), + solanaWeb3_js.u64('swapQuote2BaseFee'), + // amm vault + solanaWeb3_js.publicKey('baseVault'), + solanaWeb3_js.publicKey('quoteVault'), + // mint + solanaWeb3_js.publicKey('baseMint'), + solanaWeb3_js.publicKey('quoteMint'), + solanaWeb3_js.publicKey('lpMint'), + // market + solanaWeb3_js.publicKey('openOrders'), + solanaWeb3_js.publicKey('marketId'), + solanaWeb3_js.publicKey('marketProgramId'), + solanaWeb3_js.publicKey('targetOrders'), + solanaWeb3_js.publicKey('withdrawQueue'), + solanaWeb3_js.publicKey('lpVault'), + solanaWeb3_js.publicKey('owner'), + // true circulating supply without lock up + solanaWeb3_js.u64('lpReserve'), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 3, 'padding'), + ]); + + // CP_AMM CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C + const CPAMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(8), + solanaWeb3_js.publicKey("configId"), + solanaWeb3_js.publicKey("poolCreator"), + solanaWeb3_js.publicKey("vaultA"), + solanaWeb3_js.publicKey("vaultB"), + solanaWeb3_js.publicKey("mintLp"), + solanaWeb3_js.publicKey("mintA"), + solanaWeb3_js.publicKey("mintB"), + solanaWeb3_js.publicKey("mintProgramA"), + solanaWeb3_js.publicKey("mintProgramB"), + solanaWeb3_js.publicKey("observationId"), + solanaWeb3_js.u8("bump"), + solanaWeb3_js.u8("status"), + solanaWeb3_js.u8("lpDecimals"), + solanaWeb3_js.u8("mintDecimalA"), + solanaWeb3_js.u8("mintDecimalB"), + solanaWeb3_js.u64("lpAmount"), + solanaWeb3_js.u64("protocolFeesMintA"), + solanaWeb3_js.u64("protocolFeesMintB"), + solanaWeb3_js.u64("fundFeesMintA"), + solanaWeb3_js.u64("fundFeesMintB"), + solanaWeb3_js.u64("openTime"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 32), + ]); + + // CLMM + + const RewardInfo = solanaWeb3_js.struct([ + solanaWeb3_js.u8("rewardState"), + solanaWeb3_js.u64("openTime"), + solanaWeb3_js.u64("endTime"), + solanaWeb3_js.u64("lastUpdateTime"), + solanaWeb3_js.u128("emissionsPerSecondX64"), + solanaWeb3_js.u64("rewardTotalEmissioned"), + solanaWeb3_js.u64("rewardClaimed"), + solanaWeb3_js.publicKey("tokenMint"), + solanaWeb3_js.publicKey("tokenVault"), + solanaWeb3_js.publicKey("creator"), + solanaWeb3_js.u128("rewardGrowthGlobalX64"), + ]); + + const CLMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(8), + solanaWeb3_js.u8("bump"), + solanaWeb3_js.publicKey("ammConfig"), + solanaWeb3_js.publicKey("creator"), + solanaWeb3_js.publicKey("mintA"), + solanaWeb3_js.publicKey("mintB"), + solanaWeb3_js.publicKey("vaultA"), + solanaWeb3_js.publicKey("vaultB"), + solanaWeb3_js.publicKey("observationId"), + solanaWeb3_js.u8("mintDecimalsA"), + solanaWeb3_js.u8("mintDecimalsB"), + solanaWeb3_js.u16("tickSpacing"), + solanaWeb3_js.u128("liquidity"), + solanaWeb3_js.u128("sqrtPriceX64"), + solanaWeb3_js.i32("tickCurrent"), + solanaWeb3_js.u32(), + solanaWeb3_js.u128("feeGrowthGlobalX64A"), + solanaWeb3_js.u128("feeGrowthGlobalX64B"), + solanaWeb3_js.u64("protocolFeesTokenA"), + solanaWeb3_js.u64("protocolFeesTokenB"), + solanaWeb3_js.u128("swapInAmountTokenA"), + solanaWeb3_js.u128("swapOutAmountTokenB"), + solanaWeb3_js.u128("swapInAmountTokenB"), + solanaWeb3_js.u128("swapOutAmountTokenA"), + solanaWeb3_js.u8("status"), + solanaWeb3_js.seq(solanaWeb3_js.u8(), 7, ""), + solanaWeb3_js.seq(RewardInfo, 3, "rewardInfos"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 16, "tickArrayBitmap"), + solanaWeb3_js.u64("totalFeesTokenA"), + solanaWeb3_js.u64("totalFeesClaimedTokenA"), + solanaWeb3_js.u64("totalFeesTokenB"), + solanaWeb3_js.u64("totalFeesClaimedTokenB"), + solanaWeb3_js.u64("fundFeesTokenA"), + solanaWeb3_js.u64("fundFeesTokenB"), + solanaWeb3_js.u64("startTime"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 15 * 4 - 3, "padding"), + ]); + + function _optionalChain$4(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + + const LIQUIDITY_FEES_NUMERATOR = new solanaWeb3_js.BN(25); + const LIQUIDITY_FEES_DENOMINATOR = new solanaWeb3_js.BN(10000); + + const BNDivCeil = (bn1, bn2)=> { + const { div, mod } = bn1.divmod(bn2); + + if (mod.gt(new solanaWeb3_js.BN(0))) { + return div.add(new solanaWeb3_js.BN(1)) + } else { + return div + } + }; + + const getPairs = (base, quote)=>{ + return web3Client.request(`solana://675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8/getProgramAccounts`, { + params: { filters: [ + { dataSize: AMM_LAYOUT.span }, + { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 (Swap) + { memcmp: { offset: 400, bytes: base }}, + { memcmp: { offset: 432, bytes: quote }}, + ]}, + api: AMM_LAYOUT, + cache: 86400, // 24h, + cacheKey: ['raydium', base.toString(), quote.toString()].join('-') + }) + }; + + const getPairsWithPrice$3 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + let accounts = await getPairs(tokenIn, tokenOut); + + if(accounts.length == 0) { + accounts = await getPairs(tokenOut, tokenIn); + } + + const pairs = await Promise.all(accounts.map(async(account)=>{ + + const baseMint = account.data.baseMint.toString(); + const quoteMint = account.data.quoteMint.toString(); + + const balances = await Promise.all([ + web3Client.request(`solana://${account.data.baseVault.toString()}/getTokenAccountBalance`, { cache: 5 }), + web3Client.request(`solana://${account.data.quoteVault.toString()}/getTokenAccountBalance`, { cache: 5 }) + ]); + const baseReserve = (_optionalChain$4([balances, 'access', _ => _[0], 'optionalAccess', _2 => _2.value, 'optionalAccess', _3 => _3.amount]) ? new solanaWeb3_js.BN(_optionalChain$4([balances, 'access', _4 => _4[0], 'optionalAccess', _5 => _5.value, 'optionalAccess', _6 => _6.amount])) : new solanaWeb3_js.BN(0)).sub(account.data.baseNeedTakePnl); + const quoteReserve = (_optionalChain$4([balances, 'access', _7 => _7[1], 'optionalAccess', _8 => _8.value, 'optionalAccess', _9 => _9.amount]) ? new solanaWeb3_js.BN(_optionalChain$4([balances, 'access', _10 => _10[1], 'optionalAccess', _11 => _11.value, 'optionalAccess', _12 => _12.amount])) : new solanaWeb3_js.BN(0)).sub(account.data.quoteNeedTakePnl); + + if(baseMint === Blockchains__default['default'].solana.wrapped.address) { + if(baseReserve.lt(new solanaWeb3_js.BN(50000000))) { + return // to little liquidity + } + } else if (quoteMint === Blockchains__default['default'].solana.wrapped.address) { + if(quoteReserve.lt(new solanaWeb3_js.BN(50000000))) { + return // to little liquidity + } + } else if (Blockchains__default['default'].solana.stables.usd.includes(baseMint)) { + if((parseFloat(baseReserve.toString()) / 10 ** account.data.baseDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } else if (Blockchains__default['default'].solana.stables.usd.includes(quoteMint)) { + if((parseFloat(quoteReserve.toString()) / 10 ** account.data.quoteDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } + + const reserves = [baseReserve, quoteReserve]; + + if(tokenOut === baseMint) { + reserves.reverse(); + } + + const [reserveIn, reserveOut] = reserves; + + let price; + if(amountOut || amountOutMin) { // compute amountIn + + new solanaWeb3_js.BN(0); + let amountOutRaw = new solanaWeb3_js.BN((amountOut || amountOutMin).toString()); + + if (amountOutRaw.gt(reserveOut)) { + amountOutRaw = reserveOut.sub(new solanaWeb3_js.BN(1)); + } + + const denominator = reserveOut.sub(amountOutRaw); + const amountInWithoutFee = reserveIn.mul(amountOutRaw).div(denominator); + + price = amountInWithoutFee + .mul(LIQUIDITY_FEES_DENOMINATOR) + .div(LIQUIDITY_FEES_DENOMINATOR.sub(LIQUIDITY_FEES_NUMERATOR)) + .toString(); + + } else { // compute amountOut + + new solanaWeb3_js.BN(0); + const amountInRaw = new solanaWeb3_js.BN((amountIn || amountInMin).toString()); + const feeRaw = BNDivCeil(amountInRaw.mul(LIQUIDITY_FEES_NUMERATOR), LIQUIDITY_FEES_DENOMINATOR); + + const amountInWithFee = amountInRaw.sub(feeRaw); + const denominator = reserveIn.add(amountInWithFee); + + price = reserveOut.mul(amountInWithFee).div(denominator).toString(); + } + + return { + publicKey: account.pubkey.toString(), + baseMint, + quoteMint, + price, + data: account.data + } + + })); + + return pairs.filter(Boolean) + }; + + const getPairsWithPrice$2 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + }; + + const getPairsWithPrice$1 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + }; + + const getParisWithPriceForAllTypes = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + return Promise.all([ + getPairsWithPrice$3({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$2({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$1({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + ]).then((pairsAMM, pairsCPAMM, pairsCLMNN)=>{ + return [ + (pairsAMM || []).filter(Boolean).flat(), + (pairsCPAMM || []).filter(Boolean).flat(), + (pairsCLMNN || []).filter(Boolean).flat() + ].flat() + }) + }; + + const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + try { + return await getParisWithPriceForAllTypes({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + } catch (e) { + return [] + } + }; + + let getHighestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).gt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) + }; + + let getLowestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).lt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) + }; + + let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + + if(!pairs || pairs.length === 0) { return } + + let bestPair; + + if(amountIn || amountInMax) { + bestPair = getHighestPrice(pairs); + } else { // amount out + bestPair = getLowestPrice(pairs); + } + + return bestPair + }; + + function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + const blockchain$1 = Blockchains__default['default'].solana; + + // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. + // + // We keep 11111111111111111111111111111111 internally + // to be able to differentiate between SOL<>Token and WSOL<>Token swaps + // as they are not the same! + // + let getExchangePath$3 = ({ path }) => { + if(!path) { return } + let exchangePath = path.map((token, index) => { + if ( + token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && + path[index-1] != blockchain$1.wrapped.address + ) { + return blockchain$1.wrapped.address + } else { + return token + } + }); + + if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + exchangePath.splice(0, 1); + } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + exchangePath.splice(exchangePath.length-1, 1); + } + + return exchangePath + }; + + let pathExists$4 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + if(path.length == 1) { return false } + path = getExchangePath$3({ path }); + if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + return true + } else { + return false + } + }; + + let findPath$4 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + if( + [tokenIn, tokenOut].includes(blockchain$1.currency.address) && + [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + ) { return { path: undefined, exchangePath: undefined } } + + let path, stablesIn, stablesOut, stable; + + if (await pathExists$4({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + // direct path + path = [tokenIn, tokenOut]; + } else if ( + tokenIn != blockchain$1.wrapped.address && + tokenIn != blockchain$1.currency.address && + await pathExists$4({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$1.wrapped.address && + tokenOut != blockchain$1.currency.address && + await pathExists$4({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + ) { + // path via blockchain.wrapped.address + path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + } else if ( + !blockchain$1.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$1.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists$4({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) + ) { + // path via TOKEN_IN <> STABLE <> TOKEN_OUT + path = [tokenIn, stable, tokenOut]; + } + + // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address + // because that actually reflects how things are routed in reality: + if(_optionalChain$3([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { + path.splice(1, 0, blockchain$1.wrapped.address); + } else if(_optionalChain$3([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { + path.splice(path.length-1, 0, blockchain$1.wrapped.address); + } + return { path, exchangePath: getExchangePath$3({ path }) } + }; + + let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + + let amounts = [ethers.ethers.BigNumber.from(amountIn || amountInMax)]; + + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts + }; + + let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + + path = path.slice().reverse(); + let amounts = [ethers.ethers.BigNumber.from(amountOut || amountOutMin)]; + + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts.slice().reverse() + }; + + let getAmounts$4 = async ({ + path, + tokenIn, + tokenOut, + amountOut, + amountIn, + amountInMax, + amountOutMin + }) => { + path = getExchangePath$3({ path }); + let amounts; + if (amountOut) { + amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if (amountIn) { + amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } else if(amountOutMin) { + amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if(amountInMax) { + amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } + return { + amountOut: (amountOut || amountOutMin), + amountIn: (amountIn || amountInMax), + amountInMax: (amountInMax || amountIn), + amountOutMin: (amountOutMin || amountOut), + amounts + } + }; + + const blockchain = Blockchains__default['default'].solana; + + const getTransaction$4 = async({ + path, + amountIn, + amountInMax, + amountOut, + amountOutMin, + amounts, + amountInInput, + amountOutInput, + amountInMaxInput, + amountOutMinInput, + account + })=>{ + let transaction = { blockchain: 'solana' }; + let instructions = []; + + const exchangePath = getExchangePath$3({ path }); + if(exchangePath.length > 3) { throw 'Raydium can only handle fixed paths with a max length of 3 (2 pools)!' } + const tokenIn = exchangePath[0]; + const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; + const tokenOut = exchangePath[exchangePath.length-1]; + + let pairs; + if(exchangePath.length == 2) { + pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + } else { + if(amountInInput || amountInMaxInput) { + pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + } else { // originally amountOut + pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + } + } + + let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let wrappedAccount; + const provider = await web3Client.getProvider('solana'); + + if(startsWrapped || endsUnwrapped) { + const rent = await provider.getMinimumBalanceForRentExemption(Token__default['default'].solana.TOKEN_LAYOUT.span); + const keypair = solanaWeb3_js.Keypair.generate(); + wrappedAccount = keypair.publicKey.toString(); + const lamports = startsWrapped ? new solanaWeb3_js.BN(amountIn.toString()).add(new solanaWeb3_js.BN(rent)) : new solanaWeb3_js.BN(rent); + let createAccountInstruction = solanaWeb3_js.SystemProgram.createAccount({ + fromPubkey: new solanaWeb3_js.PublicKey(account), + newAccountPubkey: new solanaWeb3_js.PublicKey(wrappedAccount), + programId: new solanaWeb3_js.PublicKey(Token__default['default'].solana.TOKEN_PROGRAM), + space: Token__default['default'].solana.TOKEN_LAYOUT.span, + lamports + }); + createAccountInstruction.signers = [keypair]; + instructions.push(createAccountInstruction); + instructions.push( + Token__default['default'].solana.initializeAccountInstruction({ + account: wrappedAccount, + token: blockchain.wrapped.address, + owner: account + }) + ); + } + + if(pairs.length === 1) { // single hop + const tokenAccountIn = startsWrapped ? new solanaWeb3_js.PublicKey(wrappedAccount) : new solanaWeb3_js.PublicKey(await Token__default['default'].solana.findProgramAddress({ owner: account, token: tokenIn })); + const tokenAccountOut = endsUnwrapped ? new solanaWeb3_js.PublicKey(wrappedAccount) : new solanaWeb3_js.PublicKey(await Token__default['default'].solana.findProgramAddress({ owner: account, token: tokenOut })); + const pool = pairs[0]; + await web3Client.request(`solana://${pool.data.marketId}/getAccountInfo`, { api: MARKET_LAYOUT }); + + let LAYOUT, data; + LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.u8('instruction'), + solanaWeb3_js.u64('amountIn'), + solanaWeb3_js.u64('minAmountOut') + ]); + data = solanaWeb3_js.Buffer.alloc(LAYOUT.span); + LAYOUT.encode( + { + instruction: 9, + amountIn: new solanaWeb3_js.BN((amountIn || amountInMax).toString()), + minAmountOut: new solanaWeb3_js.BN((amountOutMin || amountOut).toString()), + }, + data, + ); + + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + + // let keys = [ + // // token_program + // { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // // amm + // { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // // amm authority + // { pubkey: await getAssociatedAuthority(new PublicKey(pool.publicKey)), isWritable: false, isSigner: false }, + // // amm openOrders + // { pubkey: new PublicKey(pool.data.openOrders), isWritable: true, isSigner: false }, + // // amm baseVault + // { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // // amm quoteVault + // { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // // openbook marketProgramId + // { pubkey: new PublicKey(pool.data.marketProgramId), isWritable: false, isSigner: false }, + // // openbook marketId + // { pubkey: new PublicKey(pool.data.marketId), isWritable: true, isSigner: false }, + // // openbook marketBids + // { pubkey: market.bids, isWritable: true, isSigner: false }, + // // openbook marketAsks + // { pubkey: market.asks, isWritable: true, isSigner: false }, + // // openbook eventQueue + // { pubkey: market.eventQueue, isWritable: true, isSigner: false }, + // // openbook baseVault + // { pubkey: market.baseVault, isWritable: true, isSigner: false }, + // // openbook quoteVault + // { pubkey: market.quoteVault, isWritable: true, isSigner: false }, + // // openbook marketAuthority + // { pubkey: await getAssociatedMarketAuthority(pool.data.marketProgramId, pool.data.marketId), isWritable: false, isSigner: false }, + // // user tokenAccountIn + // { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // // user tokenAccountOut + // { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // // user owner + // { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + // ] + + let keys = [ + // token_program + { pubkey: new solanaWeb3_js.PublicKey(Token__default['default'].solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // amm + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm authority + { pubkey: new solanaWeb3_js.PublicKey('5Q544fKrFoe6tsEbD7S8EmxGTJYAKtTVhAW5Q5pge4j1'), isWritable: false, isSigner: false }, + // amm openOrders + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm baseVault + { pubkey: new solanaWeb3_js.PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // amm quoteVault + { pubkey: new solanaWeb3_js.PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // openbook marketProgramId + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: false, isSigner: false }, + // openbook marketId + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketBids + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAsks + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook eventQueue + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook baseVault + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook quoteVault + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAuthority + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // user tokenAccountIn + { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // user tokenAccountOut + { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // user owner + { pubkey: new solanaWeb3_js.PublicKey(account), isWritable: true, isSigner: true } + ]; + + console.log('keys', JSON.stringify(keys)); + + instructions.push( + new solanaWeb3_js.TransactionInstruction({ + programId: new solanaWeb3_js.PublicKey('675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8'), + keys, + data, + }) + ); + } else if (pairs.length === 2) ; + + if(startsWrapped || endsUnwrapped) { + instructions.push( + Token__default['default'].solana.closeAccountInstruction({ + account: wrappedAccount, + owner: account + }) + ); + } + + // await debug(instructions, provider) + + transaction.instructions = instructions; + return transaction + }; + + var Raydium = { + findPath: findPath$4, + pathExists: pathExists$4, + getAmounts: getAmounts$4, + getTransaction: getTransaction$4, + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, + }; + + // // AMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_amm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_amm.api.span }, + // { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 + // { memcmp: { offset: 400, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 432, bytes: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_amm.api, + // }) + // console.log(accounts) + + + // // CPAMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_cpamm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_cpamm.api.span }, + // { memcmp: { offset: 168, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 200, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_cpamm.api, + // }) + // console.log(accounts) + + + // // CLMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_clmm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_clmm.api.span }, + // { memcmp: { offset: 73, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 105, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_clmm.api, + // }) + // console.log(accounts) + + const exchange$g = { + + name: 'raydium', + label: 'Raydium', + logo: '', + protocol: 'raydium', + + slippage: true, + + blockchains: ['solana'], + + solana: { + + router_amm: { + address: '675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8', + api: Raydium.AMM_LAYOUT, + }, + + router_cpamm: { + address: 'CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C', + api: Raydium.CPAMM_LAYOUT + }, + + router_clmm: { + address: 'CAMMCzo5YL8w4VFF8KVHrK22GGUsp5VTaW7grrKgrWqK', + api: Raydium.CLMM_LAYOUT + }, + } + }; + + var raydium = (scope)=>{ + return new Exchange( Object.assign(exchange$g, { scope, - findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$g }), - pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$g }), - getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$g }), + findPath: (args)=>Raydium.findPath({ ...args, exchange: exchange$g }), + pathExists: (args)=>Raydium.pathExists({ ...args, exchange: exchange$g }), + getAmounts: (args)=>Raydium.getAmounts({ ...args, exchange: exchange$g }), getPrep: (args)=>{}, - getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$g }), + getTransaction: (args)=>Raydium.getTransaction({ ...args, exchange: exchange$g }), }) ) }; @@ -4817,6 +5626,7 @@ const exchanges = [ orca(), + raydium(), uniswap_v3(), pancakeswap_v3(), uniswap_v2(), @@ -4863,6 +5673,7 @@ exchanges.solana = [ orca('solana'), + raydium('solana'), ]; exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange; }); diff --git a/dist/umd/index.solana.js b/dist/umd/index.solana.js index 0a4dbff..b59b6c3 100644 --- a/dist/umd/index.solana.js +++ b/dist/umd/index.solana.js @@ -10,7 +10,7 @@ var Blockchains__default = /*#__PURE__*/_interopDefaultLegacy(Blockchains); var Decimal__default = /*#__PURE__*/_interopDefaultLegacy(Decimal); - function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { + function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }class Route { constructor({ blockchain, tokenIn, @@ -36,10 +36,10 @@ this.decimalsOut = decimalsOut; this.path = path; this.pools = pools; - this.amountIn = _optionalChain$1([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); - this.amountOutMin = _optionalChain$1([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); - this.amountOut = _optionalChain$1([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); - this.amountInMax = _optionalChain$1([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); + this.amountIn = _optionalChain$3([amountIn, 'optionalAccess', _ => _.toString, 'call', _2 => _2()]); + this.amountOutMin = _optionalChain$3([amountOutMin, 'optionalAccess', _3 => _3.toString, 'call', _4 => _4()]); + this.amountOut = _optionalChain$3([amountOut, 'optionalAccess', _5 => _5.toString, 'call', _6 => _6()]); + this.amountInMax = _optionalChain$3([amountInMax, 'optionalAccess', _7 => _7.toString, 'call', _8 => _8()]); this.exchange = exchange; this.getPrep = getPrep; this.getTransaction = getTransaction; @@ -313,7 +313,10 @@ let amounts; // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch (e) { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { @@ -1592,7 +1595,7 @@ return accounts }; - let getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + let getPairsWithPrice$4 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { try { let accounts = await getAccounts(tokenIn, tokenOut); if(accounts.length === 0) { accounts = await getAccounts(tokenOut, tokenIn); } @@ -1619,32 +1622,31 @@ } }; - let getHighestPrice = (pairs)=>{ + let getHighestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).gt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; - let getLowestPrice = (pairs)=>{ + let getLowestPrice$1 = (pairs)=>{ return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).lt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) }; - let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { - const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + let getBestPair$1 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice$4({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); if(!pairs || pairs.length === 0) { return } let bestPair; if(amountIn || amountInMax) { - bestPair = getHighestPrice(pairs); + bestPair = getHighestPrice$1(pairs); } else { // amount out - bestPair = getLowestPrice(pairs); + bestPair = getLowestPrice$1(pairs); } - return bestPair }; - function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } - const blockchain$1 = Blockchains__default['default'].solana; + function _optionalChain$2(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + const blockchain$3 = Blockchains__default['default'].solana; // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. // @@ -1652,64 +1654,64 @@ // to be able to differentiate between SOL<>Token and WSOL<>Token swaps // as they are not the same! // - let getExchangePath = ({ path }) => { + let getExchangePath$1 = ({ path }) => { if(!path) { return } let exchangePath = path.map((token, index) => { if ( - token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && - path[index-1] != blockchain$1.wrapped.address + token === blockchain$3.currency.address && path[index+1] != blockchain$3.wrapped.address && + path[index-1] != blockchain$3.wrapped.address ) { - return blockchain$1.wrapped.address + return blockchain$3.wrapped.address } else { return token } }); - if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + if(exchangePath[0] == blockchain$3.currency.address && exchangePath[1] == blockchain$3.wrapped.address) { exchangePath.splice(0, 1); - } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + } else if(exchangePath[exchangePath.length-1] == blockchain$3.currency.address && exchangePath[exchangePath.length-2] == blockchain$3.wrapped.address) { exchangePath.splice(exchangePath.length-1, 1); } return exchangePath }; - let pathExists = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + let pathExists$1 = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { if(path.length == 1) { return false } - path = getExchangePath({ path }); - if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + path = getExchangePath$1({ path }); + if((await getPairsWithPrice$4({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { return true } else { return false } }; - let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + let findPath$1 = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { if( - [tokenIn, tokenOut].includes(blockchain$1.currency.address) && - [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + [tokenIn, tokenOut].includes(blockchain$3.currency.address) && + [tokenIn, tokenOut].includes(blockchain$3.wrapped.address) ) { return { path: undefined, exchangePath: undefined } } let path, stablesIn, stablesOut, stable; - if (await pathExists({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + if (await pathExists$1({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { // direct path path = [tokenIn, tokenOut]; } else if ( - tokenIn != blockchain$1.wrapped.address && - tokenIn != blockchain$1.currency.address && - await pathExists({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && - tokenOut != blockchain$1.wrapped.address && - tokenOut != blockchain$1.currency.address && - await pathExists({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + tokenIn != blockchain$3.wrapped.address && + tokenIn != blockchain$3.currency.address && + await pathExists$1({ path: [tokenIn, blockchain$3.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$3.wrapped.address && + tokenOut != blockchain$3.currency.address && + await pathExists$1({ path: [tokenOut, blockchain$3.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ) { // path via blockchain.wrapped.address - path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + path = [tokenIn, blockchain$3.wrapped.address, tokenOut]; } else if ( - !blockchain$1.stables.usd.includes(tokenIn) && - (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && - !blockchain$1.stables.usd.includes(tokenOut) && - (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$1({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$3.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$3.stables.usd.map(async(stable)=>await pathExists$1({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) ) { // path via TOKEN_IN <> STABLE <> TOKEN_OUT @@ -1718,22 +1720,26 @@ // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address // because that actually reflects how things are routed in reality: - if(_optionalChain([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { - path.splice(1, 0, blockchain$1.wrapped.address); - } else if(_optionalChain([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { - path.splice(path.length-1, 0, blockchain$1.wrapped.address); + if(_optionalChain$2([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$3.currency.address) { + path.splice(1, 0, blockchain$3.wrapped.address); + } else if(_optionalChain$2([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$3.currency.address) { + path.splice(path.length-1, 0, blockchain$3.wrapped.address); } - return { path, exchangePath: getExchangePath({ path }) } + return { path, exchangePath: getExchangePath$1({ path }) } }; - let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + let getAmountsOut$1 = async ({ path, amountIn, amountInMax }) => { let amounts = [ethers.ethers.BigNumber.from(amountIn || amountInMax)]; - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -1741,15 +1747,19 @@ return amounts }; - let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + let getAmountsIn$1 = async({ path, amountOut, amountOutMin }) => { path = path.slice().reverse(); let amounts = [ethers.ethers.BigNumber.from(amountOut || amountOutMin)]; - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); if (path.length === 3) { - amounts.push(ethers.ethers.BigNumber.from((await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined })).price)); + let bestPair = await getBestPair$1({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); } if(amounts.length != path.length) { return } @@ -1757,7 +1767,7 @@ return amounts.slice().reverse() }; - let getAmounts = async ({ + let getAmounts$1 = async ({ path, tokenIn, tokenOut, @@ -1766,10 +1776,10 @@ amountInMax, amountOutMin }) => { - path = getExchangePath({ path }); + path = getExchangePath$1({ path }); let amounts; if (amountOut) { - amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOut, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -1777,7 +1787,7 @@ amountInMax = amountIn; } } else if (amountIn) { - amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountIn, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -1785,7 +1795,7 @@ amountOutMin = amountOut; } } else if(amountOutMin) { - amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amounts = await getAmountsIn$1({ path, amountOutMin, tokenIn, tokenOut }); amountIn = amounts ? amounts[0] : undefined; if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { return {} @@ -1793,7 +1803,7 @@ amountInMax = amountIn; } } else if(amountInMax) { - amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amounts = await getAmountsOut$1({ path, amountInMax, tokenIn, tokenOut }); amountOut = amounts ? amounts[amounts.length-1] : undefined; if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { return {} @@ -1810,7 +1820,7 @@ } }; - const blockchain = Blockchains__default['default'].solana; + const blockchain$2 = Blockchains__default['default'].solana; const SWAP_INSTRUCTION = new solanaWeb3_js.BN("14449647541112719096"); const TWO_HOP_SWAP_INSTRUCTION = new solanaWeb3_js.BN("16635068063392030915"); @@ -2025,7 +2035,7 @@ return data }; - const getTransaction = async ({ + const getTransaction$1 = async ({ path, amountIn, amountInMax, @@ -2041,7 +2051,7 @@ let transaction = { blockchain: 'solana' }; let instructions = []; - const exchangePath = getExchangePath({ path }); + const exchangePath = getExchangePath$1({ path }); if(exchangePath.length > 3) { throw 'Orca can only handle fixed paths with a max length of 3 (2 pools)!' } const tokenIn = exchangePath[0]; const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; @@ -2049,19 +2059,19 @@ let pairs; if(exchangePath.length == 2) { - pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + pairs = [await getBestPair$1({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; } else { if(amountInInput || amountInMaxInput) { - pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; - pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); } else { // originally amountOut - pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; - pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + pairs = [await getBestPair$1({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair$1({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); } } - let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); - let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let startsWrapped = (path[0] === blockchain$2.currency.address && exchangePath[0] === blockchain$2.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain$2.currency.address && exchangePath[exchangePath.length-1] === blockchain$2.wrapped.address); let wrappedAccount; const provider = await web3ClientSolana.getProvider('solana'); @@ -2082,7 +2092,7 @@ instructions.push( Token__default['default'].solana.initializeAccountInstruction({ account: wrappedAccount, - token: blockchain.wrapped.address, + token: blockchain$2.wrapped.address, owner: account }) ); @@ -2179,14 +2189,14 @@ }; var Orca = { - findPath, - pathExists, - getAmounts, - getTransaction, + findPath: findPath$1, + pathExists: pathExists$1, + getAmounts: getAmounts$1, + getTransaction: getTransaction$1, WHIRLPOOL_LAYOUT, }; - const exchange = { + const exchange$1 = { name: 'orca', label: 'Orca', @@ -2207,22 +2217,822 @@ var orca = (scope)=>{ + return new Exchange( + + Object.assign(exchange$1, { + scope, + + findPath: (args)=>Orca.findPath({ ...args, exchange: exchange$1 }), + pathExists: (args)=>Orca.pathExists({ ...args, exchange: exchange$1 }), + getAmounts: (args)=>Orca.getAmounts({ ...args, exchange: exchange$1 }), + getPrep: (args)=>{}, + getTransaction: (args)=>Orca.getTransaction({ ...args, exchange: exchange$1 }), + }) + ) + }; + + // OpenBook Market + const MARKET_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(5), + solanaWeb3_js.blob(8), // accountFlagsLayout('accountFlags'), + solanaWeb3_js.publicKey('ownAddress'), + solanaWeb3_js.u64('vaultSignerNonce'), + solanaWeb3_js.publicKey('baseMint'), + solanaWeb3_js.publicKey('quoteMint'), + solanaWeb3_js.publicKey('baseVault'), + solanaWeb3_js.u64('baseDepositsTotal'), + solanaWeb3_js.u64('baseFeesAccrued'), + solanaWeb3_js.publicKey('quoteVault'), + solanaWeb3_js.u64('quoteDepositsTotal'), + solanaWeb3_js.u64('quoteFeesAccrued'), + solanaWeb3_js.u64('quoteDustThreshold'), + solanaWeb3_js.publicKey('requestQueue'), + solanaWeb3_js.publicKey('eventQueue'), + solanaWeb3_js.publicKey('bids'), + solanaWeb3_js.publicKey('asks'), + solanaWeb3_js.u64('baseLotSize'), + solanaWeb3_js.u64('quoteLotSize'), + solanaWeb3_js.u64('feeRateBps'), + solanaWeb3_js.u64('referrerRebatesAccrued'), + solanaWeb3_js.blob(7), + ]); + + // AMM 675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8 + const AMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.u64('status'), + solanaWeb3_js.u64('nonce'), + solanaWeb3_js.u64('maxOrder'), + solanaWeb3_js.u64('depth'), + solanaWeb3_js.u64('baseDecimal'), + solanaWeb3_js.u64('quoteDecimal'), + solanaWeb3_js.u64('state'), + solanaWeb3_js.u64('resetFlag'), + solanaWeb3_js.u64('minSize'), + solanaWeb3_js.u64('volMaxCutRatio'), + solanaWeb3_js.u64('amountWaveRatio'), + solanaWeb3_js.u64('baseLotSize'), + solanaWeb3_js.u64('quoteLotSize'), + solanaWeb3_js.u64('minPriceMultiplier'), + solanaWeb3_js.u64('maxPriceMultiplier'), + solanaWeb3_js.u64('systemDecimalValue'), + solanaWeb3_js.u64('minSeparateNumerator'), + solanaWeb3_js.u64('minSeparateDenominator'), + solanaWeb3_js.u64('tradeFeeNumerator'), + solanaWeb3_js.u64('tradeFeeDenominator'), + solanaWeb3_js.u64('pnlNumerator'), + solanaWeb3_js.u64('pnlDenominator'), + solanaWeb3_js.u64('swapFeeNumerator'), + solanaWeb3_js.u64('swapFeeDenominator'), + solanaWeb3_js.u64('baseNeedTakePnl'), + solanaWeb3_js.u64('quoteNeedTakePnl'), + solanaWeb3_js.u64('quoteTotalPnl'), + solanaWeb3_js.u64('baseTotalPnl'), + solanaWeb3_js.u64('poolOpenTime'), + solanaWeb3_js.u64('punishPcAmount'), + solanaWeb3_js.u64('punishCoinAmount'), + solanaWeb3_js.u64('orderbookToInitTime'), + solanaWeb3_js.u128('swapBaseInAmount'), + solanaWeb3_js.u128('swapQuoteOutAmount'), + solanaWeb3_js.u64('swapBase2QuoteFee'), + solanaWeb3_js.u128('swapQuoteInAmount'), + solanaWeb3_js.u128('swapBaseOutAmount'), + solanaWeb3_js.u64('swapQuote2BaseFee'), + // amm vault + solanaWeb3_js.publicKey('baseVault'), + solanaWeb3_js.publicKey('quoteVault'), + // mint + solanaWeb3_js.publicKey('baseMint'), + solanaWeb3_js.publicKey('quoteMint'), + solanaWeb3_js.publicKey('lpMint'), + // market + solanaWeb3_js.publicKey('openOrders'), + solanaWeb3_js.publicKey('marketId'), + solanaWeb3_js.publicKey('marketProgramId'), + solanaWeb3_js.publicKey('targetOrders'), + solanaWeb3_js.publicKey('withdrawQueue'), + solanaWeb3_js.publicKey('lpVault'), + solanaWeb3_js.publicKey('owner'), + // true circulating supply without lock up + solanaWeb3_js.u64('lpReserve'), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 3, 'padding'), + ]); + + // CP_AMM CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C + const CPAMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(8), + solanaWeb3_js.publicKey("configId"), + solanaWeb3_js.publicKey("poolCreator"), + solanaWeb3_js.publicKey("vaultA"), + solanaWeb3_js.publicKey("vaultB"), + solanaWeb3_js.publicKey("mintLp"), + solanaWeb3_js.publicKey("mintA"), + solanaWeb3_js.publicKey("mintB"), + solanaWeb3_js.publicKey("mintProgramA"), + solanaWeb3_js.publicKey("mintProgramB"), + solanaWeb3_js.publicKey("observationId"), + solanaWeb3_js.u8("bump"), + solanaWeb3_js.u8("status"), + solanaWeb3_js.u8("lpDecimals"), + solanaWeb3_js.u8("mintDecimalA"), + solanaWeb3_js.u8("mintDecimalB"), + solanaWeb3_js.u64("lpAmount"), + solanaWeb3_js.u64("protocolFeesMintA"), + solanaWeb3_js.u64("protocolFeesMintB"), + solanaWeb3_js.u64("fundFeesMintA"), + solanaWeb3_js.u64("fundFeesMintB"), + solanaWeb3_js.u64("openTime"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 32), + ]); + + // CLMM + + const RewardInfo = solanaWeb3_js.struct([ + solanaWeb3_js.u8("rewardState"), + solanaWeb3_js.u64("openTime"), + solanaWeb3_js.u64("endTime"), + solanaWeb3_js.u64("lastUpdateTime"), + solanaWeb3_js.u128("emissionsPerSecondX64"), + solanaWeb3_js.u64("rewardTotalEmissioned"), + solanaWeb3_js.u64("rewardClaimed"), + solanaWeb3_js.publicKey("tokenMint"), + solanaWeb3_js.publicKey("tokenVault"), + solanaWeb3_js.publicKey("creator"), + solanaWeb3_js.u128("rewardGrowthGlobalX64"), + ]); + + const CLMM_LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.blob(8), + solanaWeb3_js.u8("bump"), + solanaWeb3_js.publicKey("ammConfig"), + solanaWeb3_js.publicKey("creator"), + solanaWeb3_js.publicKey("mintA"), + solanaWeb3_js.publicKey("mintB"), + solanaWeb3_js.publicKey("vaultA"), + solanaWeb3_js.publicKey("vaultB"), + solanaWeb3_js.publicKey("observationId"), + solanaWeb3_js.u8("mintDecimalsA"), + solanaWeb3_js.u8("mintDecimalsB"), + solanaWeb3_js.u16("tickSpacing"), + solanaWeb3_js.u128("liquidity"), + solanaWeb3_js.u128("sqrtPriceX64"), + solanaWeb3_js.i32("tickCurrent"), + solanaWeb3_js.u32(), + solanaWeb3_js.u128("feeGrowthGlobalX64A"), + solanaWeb3_js.u128("feeGrowthGlobalX64B"), + solanaWeb3_js.u64("protocolFeesTokenA"), + solanaWeb3_js.u64("protocolFeesTokenB"), + solanaWeb3_js.u128("swapInAmountTokenA"), + solanaWeb3_js.u128("swapOutAmountTokenB"), + solanaWeb3_js.u128("swapInAmountTokenB"), + solanaWeb3_js.u128("swapOutAmountTokenA"), + solanaWeb3_js.u8("status"), + solanaWeb3_js.seq(solanaWeb3_js.u8(), 7, ""), + solanaWeb3_js.seq(RewardInfo, 3, "rewardInfos"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 16, "tickArrayBitmap"), + solanaWeb3_js.u64("totalFeesTokenA"), + solanaWeb3_js.u64("totalFeesClaimedTokenA"), + solanaWeb3_js.u64("totalFeesTokenB"), + solanaWeb3_js.u64("totalFeesClaimedTokenB"), + solanaWeb3_js.u64("fundFeesTokenA"), + solanaWeb3_js.u64("fundFeesTokenB"), + solanaWeb3_js.u64("startTime"), + solanaWeb3_js.seq(solanaWeb3_js.u64(), 15 * 4 - 3, "padding"), + ]); + + function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + + const LIQUIDITY_FEES_NUMERATOR = new solanaWeb3_js.BN(25); + const LIQUIDITY_FEES_DENOMINATOR = new solanaWeb3_js.BN(10000); + + const BNDivCeil = (bn1, bn2)=> { + const { div, mod } = bn1.divmod(bn2); + + if (mod.gt(new solanaWeb3_js.BN(0))) { + return div.add(new solanaWeb3_js.BN(1)) + } else { + return div + } + }; + + const getPairs = (base, quote)=>{ + return web3ClientSolana.request(`solana://675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8/getProgramAccounts`, { + params: { filters: [ + { dataSize: AMM_LAYOUT.span }, + { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 (Swap) + { memcmp: { offset: 400, bytes: base }}, + { memcmp: { offset: 432, bytes: quote }}, + ]}, + api: AMM_LAYOUT, + cache: 86400, // 24h, + cacheKey: ['raydium', base.toString(), quote.toString()].join('-') + }) + }; + + const getPairsWithPrice$3 = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + let accounts = await getPairs(tokenIn, tokenOut); + + if(accounts.length == 0) { + accounts = await getPairs(tokenOut, tokenIn); + } + + const pairs = await Promise.all(accounts.map(async(account)=>{ + + const baseMint = account.data.baseMint.toString(); + const quoteMint = account.data.quoteMint.toString(); + + const balances = await Promise.all([ + web3ClientSolana.request(`solana://${account.data.baseVault.toString()}/getTokenAccountBalance`, { cache: 5 }), + web3ClientSolana.request(`solana://${account.data.quoteVault.toString()}/getTokenAccountBalance`, { cache: 5 }) + ]); + const baseReserve = (_optionalChain$1([balances, 'access', _ => _[0], 'optionalAccess', _2 => _2.value, 'optionalAccess', _3 => _3.amount]) ? new solanaWeb3_js.BN(_optionalChain$1([balances, 'access', _4 => _4[0], 'optionalAccess', _5 => _5.value, 'optionalAccess', _6 => _6.amount])) : new solanaWeb3_js.BN(0)).sub(account.data.baseNeedTakePnl); + const quoteReserve = (_optionalChain$1([balances, 'access', _7 => _7[1], 'optionalAccess', _8 => _8.value, 'optionalAccess', _9 => _9.amount]) ? new solanaWeb3_js.BN(_optionalChain$1([balances, 'access', _10 => _10[1], 'optionalAccess', _11 => _11.value, 'optionalAccess', _12 => _12.amount])) : new solanaWeb3_js.BN(0)).sub(account.data.quoteNeedTakePnl); + + if(baseMint === Blockchains__default['default'].solana.wrapped.address) { + if(baseReserve.lt(new solanaWeb3_js.BN(50000000))) { + return // to little liquidity + } + } else if (quoteMint === Blockchains__default['default'].solana.wrapped.address) { + if(quoteReserve.lt(new solanaWeb3_js.BN(50000000))) { + return // to little liquidity + } + } else if (Blockchains__default['default'].solana.stables.usd.includes(baseMint)) { + if((parseFloat(baseReserve.toString()) / 10 ** account.data.baseDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } else if (Blockchains__default['default'].solana.stables.usd.includes(quoteMint)) { + if((parseFloat(quoteReserve.toString()) / 10 ** account.data.quoteDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } + + const reserves = [baseReserve, quoteReserve]; + + if(tokenOut === baseMint) { + reserves.reverse(); + } + + const [reserveIn, reserveOut] = reserves; + + let price; + if(amountOut || amountOutMin) { // compute amountIn + + new solanaWeb3_js.BN(0); + let amountOutRaw = new solanaWeb3_js.BN((amountOut || amountOutMin).toString()); + + if (amountOutRaw.gt(reserveOut)) { + amountOutRaw = reserveOut.sub(new solanaWeb3_js.BN(1)); + } + + const denominator = reserveOut.sub(amountOutRaw); + const amountInWithoutFee = reserveIn.mul(amountOutRaw).div(denominator); + + price = amountInWithoutFee + .mul(LIQUIDITY_FEES_DENOMINATOR) + .div(LIQUIDITY_FEES_DENOMINATOR.sub(LIQUIDITY_FEES_NUMERATOR)) + .toString(); + + } else { // compute amountOut + + new solanaWeb3_js.BN(0); + const amountInRaw = new solanaWeb3_js.BN((amountIn || amountInMin).toString()); + const feeRaw = BNDivCeil(amountInRaw.mul(LIQUIDITY_FEES_NUMERATOR), LIQUIDITY_FEES_DENOMINATOR); + + const amountInWithFee = amountInRaw.sub(feeRaw); + const denominator = reserveIn.add(amountInWithFee); + + price = reserveOut.mul(amountInWithFee).div(denominator).toString(); + } + + return { + publicKey: account.pubkey.toString(), + baseMint, + quoteMint, + price, + data: account.data + } + + })); + + return pairs.filter(Boolean) + }; + + const getPairsWithPrice$2 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + }; + + const getPairsWithPrice$1 = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + }; + + const getParisWithPriceForAllTypes = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + return Promise.all([ + getPairsWithPrice$3({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$2({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPrice$1({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + ]).then((pairsAMM, pairsCPAMM, pairsCLMNN)=>{ + return [ + (pairsAMM || []).filter(Boolean).flat(), + (pairsCPAMM || []).filter(Boolean).flat(), + (pairsCLMNN || []).filter(Boolean).flat() + ].flat() + }) + }; + + const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + try { + return await getParisWithPriceForAllTypes({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + } catch (e) { + return [] + } + }; + + let getHighestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).gt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) + }; + + let getLowestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.ethers.BigNumber.from(currentPair.price).lt(ethers.ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) + }; + + let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }); + + if(!pairs || pairs.length === 0) { return } + + let bestPair; + + if(amountIn || amountInMax) { + bestPair = getHighestPrice(pairs); + } else { // amount out + bestPair = getLowestPrice(pairs); + } + + return bestPair + }; + + function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } + const blockchain$1 = Blockchains__default['default'].solana; + + // Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. + // + // We keep 11111111111111111111111111111111 internally + // to be able to differentiate between SOL<>Token and WSOL<>Token swaps + // as they are not the same! + // + let getExchangePath = ({ path }) => { + if(!path) { return } + let exchangePath = path.map((token, index) => { + if ( + token === blockchain$1.currency.address && path[index+1] != blockchain$1.wrapped.address && + path[index-1] != blockchain$1.wrapped.address + ) { + return blockchain$1.wrapped.address + } else { + return token + } + }); + + if(exchangePath[0] == blockchain$1.currency.address && exchangePath[1] == blockchain$1.wrapped.address) { + exchangePath.splice(0, 1); + } else if(exchangePath[exchangePath.length-1] == blockchain$1.currency.address && exchangePath[exchangePath.length-2] == blockchain$1.wrapped.address) { + exchangePath.splice(exchangePath.length-1, 1); + } + + return exchangePath + }; + + let pathExists = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + if(path.length == 1) { return false } + path = getExchangePath({ path }); + if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + return true + } else { + return false + } + }; + + let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + if( + [tokenIn, tokenOut].includes(blockchain$1.currency.address) && + [tokenIn, tokenOut].includes(blockchain$1.wrapped.address) + ) { return { path: undefined, exchangePath: undefined } } + + let path, stablesIn, stablesOut, stable; + + if (await pathExists({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + // direct path + path = [tokenIn, tokenOut]; + } else if ( + tokenIn != blockchain$1.wrapped.address && + tokenIn != blockchain$1.currency.address && + await pathExists({ path: [tokenIn, blockchain$1.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain$1.wrapped.address && + tokenOut != blockchain$1.currency.address && + await pathExists({ path: [tokenOut, blockchain$1.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + ) { + // path via blockchain.wrapped.address + path = [tokenIn, blockchain$1.wrapped.address, tokenOut]; + } else if ( + !blockchain$1.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain$1.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain$1.stables.usd.map(async(stable)=>await pathExists({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) + ) { + // path via TOKEN_IN <> STABLE <> TOKEN_OUT + path = [tokenIn, stable, tokenOut]; + } + + // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address + // because that actually reflects how things are routed in reality: + if(_optionalChain([path, 'optionalAccess', _ => _.length]) && path[0] == blockchain$1.currency.address) { + path.splice(1, 0, blockchain$1.wrapped.address); + } else if(_optionalChain([path, 'optionalAccess', _2 => _2.length]) && path[path.length-1] == blockchain$1.currency.address) { + path.splice(path.length-1, 0, blockchain$1.wrapped.address); + } + return { path, exchangePath: getExchangePath({ path }) } + }; + + let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + + let amounts = [ethers.ethers.BigNumber.from(amountIn || amountInMax)]; + + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts + }; + + let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + + path = path.slice().reverse(); + let amounts = [ethers.ethers.BigNumber.from(amountOut || amountOutMin)]; + + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }); + if(!bestPair){ return } + amounts.push(ethers.ethers.BigNumber.from(bestPair.price)); + } + + if(amounts.length != path.length) { return } + + return amounts.slice().reverse() + }; + + let getAmounts = async ({ + path, + tokenIn, + tokenOut, + amountOut, + amountIn, + amountInMax, + amountOutMin + }) => { + path = getExchangePath({ path }); + let amounts; + if (amountOut) { + amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if (amountIn) { + amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } else if(amountOutMin) { + amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }); + amountIn = amounts ? amounts[0] : undefined; + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn; + } + } else if(amountInMax) { + amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }); + amountOut = amounts ? amounts[amounts.length-1] : undefined; + if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut; + } + } + return { + amountOut: (amountOut || amountOutMin), + amountIn: (amountIn || amountInMax), + amountInMax: (amountInMax || amountIn), + amountOutMin: (amountOutMin || amountOut), + amounts + } + }; + + const blockchain = Blockchains__default['default'].solana; + + const getTransaction = async({ + path, + amountIn, + amountInMax, + amountOut, + amountOutMin, + amounts, + amountInInput, + amountOutInput, + amountInMaxInput, + amountOutMinInput, + account + })=>{ + let transaction = { blockchain: 'solana' }; + let instructions = []; + + const exchangePath = getExchangePath({ path }); + if(exchangePath.length > 3) { throw 'Raydium can only handle fixed paths with a max length of 3 (2 pools)!' } + const tokenIn = exchangePath[0]; + const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined; + const tokenOut = exchangePath[exchangePath.length-1]; + + let pairs; + if(exchangePath.length == 2) { + pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })]; + } else { + if(amountInInput || amountInMaxInput) { + pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })]; + pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })); + } else { // originally amountOut + pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })]; + pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })); + } + } + + let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address); + let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address); + let wrappedAccount; + const provider = await web3ClientSolana.getProvider('solana'); + + if(startsWrapped || endsUnwrapped) { + const rent = await provider.getMinimumBalanceForRentExemption(Token__default['default'].solana.TOKEN_LAYOUT.span); + const keypair = solanaWeb3_js.Keypair.generate(); + wrappedAccount = keypair.publicKey.toString(); + const lamports = startsWrapped ? new solanaWeb3_js.BN(amountIn.toString()).add(new solanaWeb3_js.BN(rent)) : new solanaWeb3_js.BN(rent); + let createAccountInstruction = solanaWeb3_js.SystemProgram.createAccount({ + fromPubkey: new solanaWeb3_js.PublicKey(account), + newAccountPubkey: new solanaWeb3_js.PublicKey(wrappedAccount), + programId: new solanaWeb3_js.PublicKey(Token__default['default'].solana.TOKEN_PROGRAM), + space: Token__default['default'].solana.TOKEN_LAYOUT.span, + lamports + }); + createAccountInstruction.signers = [keypair]; + instructions.push(createAccountInstruction); + instructions.push( + Token__default['default'].solana.initializeAccountInstruction({ + account: wrappedAccount, + token: blockchain.wrapped.address, + owner: account + }) + ); + } + + if(pairs.length === 1) { // single hop + const tokenAccountIn = startsWrapped ? new solanaWeb3_js.PublicKey(wrappedAccount) : new solanaWeb3_js.PublicKey(await Token__default['default'].solana.findProgramAddress({ owner: account, token: tokenIn })); + const tokenAccountOut = endsUnwrapped ? new solanaWeb3_js.PublicKey(wrappedAccount) : new solanaWeb3_js.PublicKey(await Token__default['default'].solana.findProgramAddress({ owner: account, token: tokenOut })); + const pool = pairs[0]; + await web3ClientSolana.request(`solana://${pool.data.marketId}/getAccountInfo`, { api: MARKET_LAYOUT }); + + let LAYOUT, data; + LAYOUT = solanaWeb3_js.struct([ + solanaWeb3_js.u8('instruction'), + solanaWeb3_js.u64('amountIn'), + solanaWeb3_js.u64('minAmountOut') + ]); + data = solanaWeb3_js.Buffer.alloc(LAYOUT.span); + LAYOUT.encode( + { + instruction: 9, + amountIn: new solanaWeb3_js.BN((amountIn || amountInMax).toString()), + minAmountOut: new solanaWeb3_js.BN((amountOutMin || amountOut).toString()), + }, + data, + ); + + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + + // let keys = [ + // // token_program + // { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // // amm + // { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // // amm authority + // { pubkey: await getAssociatedAuthority(new PublicKey(pool.publicKey)), isWritable: false, isSigner: false }, + // // amm openOrders + // { pubkey: new PublicKey(pool.data.openOrders), isWritable: true, isSigner: false }, + // // amm baseVault + // { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // // amm quoteVault + // { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // // openbook marketProgramId + // { pubkey: new PublicKey(pool.data.marketProgramId), isWritable: false, isSigner: false }, + // // openbook marketId + // { pubkey: new PublicKey(pool.data.marketId), isWritable: true, isSigner: false }, + // // openbook marketBids + // { pubkey: market.bids, isWritable: true, isSigner: false }, + // // openbook marketAsks + // { pubkey: market.asks, isWritable: true, isSigner: false }, + // // openbook eventQueue + // { pubkey: market.eventQueue, isWritable: true, isSigner: false }, + // // openbook baseVault + // { pubkey: market.baseVault, isWritable: true, isSigner: false }, + // // openbook quoteVault + // { pubkey: market.quoteVault, isWritable: true, isSigner: false }, + // // openbook marketAuthority + // { pubkey: await getAssociatedMarketAuthority(pool.data.marketProgramId, pool.data.marketId), isWritable: false, isSigner: false }, + // // user tokenAccountIn + // { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // // user tokenAccountOut + // { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // // user owner + // { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + // ] + + let keys = [ + // token_program + { pubkey: new solanaWeb3_js.PublicKey(Token__default['default'].solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // amm + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm authority + { pubkey: new solanaWeb3_js.PublicKey('5Q544fKrFoe6tsEbD7S8EmxGTJYAKtTVhAW5Q5pge4j1'), isWritable: false, isSigner: false }, + // amm openOrders + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm baseVault + { pubkey: new solanaWeb3_js.PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // amm quoteVault + { pubkey: new solanaWeb3_js.PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // openbook marketProgramId + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: false, isSigner: false }, + // openbook marketId + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketBids + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAsks + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook eventQueue + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook baseVault + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook quoteVault + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAuthority + { pubkey: new solanaWeb3_js.PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // user tokenAccountIn + { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // user tokenAccountOut + { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // user owner + { pubkey: new solanaWeb3_js.PublicKey(account), isWritable: true, isSigner: true } + ]; + + console.log('keys', JSON.stringify(keys)); + + instructions.push( + new solanaWeb3_js.TransactionInstruction({ + programId: new solanaWeb3_js.PublicKey('675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8'), + keys, + data, + }) + ); + } else if (pairs.length === 2) ; + + if(startsWrapped || endsUnwrapped) { + instructions.push( + Token__default['default'].solana.closeAccountInstruction({ + account: wrappedAccount, + owner: account + }) + ); + } + + // await debug(instructions, provider) + + transaction.instructions = instructions; + return transaction + }; + + var Raydium = { + findPath, + pathExists, + getAmounts, + getTransaction, + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, + }; + + // // AMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_amm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_amm.api.span }, + // { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 + // { memcmp: { offset: 400, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 432, bytes: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_amm.api, + // }) + // console.log(accounts) + + + // // CPAMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_cpamm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_cpamm.api.span }, + // { memcmp: { offset: 168, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 200, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_cpamm.api, + // }) + // console.log(accounts) + + + // // CLMM + // let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_clmm.address}/getProgramAccounts`, { + // params: { filters: [ + // { dataSize: Web3Exchanges.raydium.solana.router_clmm.api.span }, + // { memcmp: { offset: 73, bytes: "So11111111111111111111111111111111111111112" }}, + // { memcmp: { offset: 105, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, + // ]}, + // api: Web3Exchanges.raydium.solana.router_clmm.api, + // }) + // console.log(accounts) + + const exchange = { + + name: 'raydium', + label: 'Raydium', + logo: '', + protocol: 'raydium', + + slippage: true, + + blockchains: ['solana'], + + solana: { + + router_amm: { + address: '675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8', + api: Raydium.AMM_LAYOUT, + }, + + router_cpamm: { + address: 'CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C', + api: Raydium.CPAMM_LAYOUT + }, + + router_clmm: { + address: 'CAMMCzo5YL8w4VFF8KVHrK22GGUsp5VTaW7grrKgrWqK', + api: Raydium.CLMM_LAYOUT + }, + } + }; + + var raydium = (scope)=>{ + return new Exchange( Object.assign(exchange, { scope, - findPath: (args)=>Orca.findPath({ ...args, exchange }), - pathExists: (args)=>Orca.pathExists({ ...args, exchange }), - getAmounts: (args)=>Orca.getAmounts({ ...args, exchange }), + findPath: (args)=>Raydium.findPath({ ...args, exchange }), + pathExists: (args)=>Raydium.pathExists({ ...args, exchange }), + getAmounts: (args)=>Raydium.getAmounts({ ...args, exchange }), getPrep: (args)=>{}, - getTransaction: (args)=>Orca.getTransaction({ ...args, exchange }), + getTransaction: (args)=>Raydium.getTransaction({ ...args, exchange }), }) ) }; const exchanges = [ orca(), + raydium(), ]; exchanges.forEach((exchange)=>{ exchanges[exchange.name] = exchange; @@ -2230,6 +3040,7 @@ exchanges.solana = [ orca('solana'), + raydium('solana'), ]; exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange; }); diff --git a/examples/raydium.md b/examples/raydium.md new file mode 100644 index 0000000..6b75da8 --- /dev/null +++ b/examples/raydium.md @@ -0,0 +1,20 @@ +## AMM v4 (Legacy) + +### 1 Pool + +```javascript +let route = await Web3Exchanges.raydium.route({ + blockchain: 'solana', + tokenIn: 'EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v', + tokenOut: 'HhJpBhRRn4g56VsyLuT8DL5Bv31HkXqsrahTTUCZeZg4', + amountOutMin: 0.01 +}) + +let wallets = await Web3Wallets.getWallets() +let wallet = wallets[1] +let account = await wallet.account() +let transaction = await route.getTransaction({ account }) + +wallet.sendTransaction(transaction) +``` + diff --git a/package.evm.json b/package.evm.json index 9d312b4..9733e64 100644 --- a/package.evm.json +++ b/package.evm.json @@ -1,7 +1,7 @@ { "name": "@depay/web3-exchanges-evm", "moduleName": "Web3Exchanges", - "version": "13.13.1", + "version": "13.15.0", "description": "JavaScript library simplifying decentralized web3 exchange routing for multiple blockchains and exchanges.", "main": "dist/umd/index.evm.js", "module": "dist/esm/index.evm.js", diff --git a/package.json b/package.json index 3e4053e..f9bb210 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@depay/web3-exchanges", "moduleName": "Web3Exchanges", - "version": "13.13.1", + "version": "13.15.0", "description": "JavaScript library simplifying decentralized web3 exchange routing for multiple blockchains and exchanges.", "main": "dist/umd/index.js", "module": "dist/esm/index.js", diff --git a/package.solana.json b/package.solana.json index a994af3..26e0f10 100644 --- a/package.solana.json +++ b/package.solana.json @@ -1,7 +1,7 @@ { "name": "@depay/web3-exchanges-solana", "moduleName": "Web3Exchanges", - "version": "13.13.1", + "version": "13.15.0", "description": "JavaScript library simplifying decentralized web3 exchange routing for multiple blockchains and exchanges.", "main": "dist/umd/index.solana.js", "module": "dist/esm/index.solana.js", diff --git a/src/classes/Exchange.js b/src/classes/Exchange.js index 9ae7fa5..1edc512 100644 --- a/src/classes/Exchange.js +++ b/src/classes/Exchange.js @@ -47,7 +47,10 @@ const route = ({ let amounts // includes intermediary amounts for longer routes try { ;({ amountIn, amountInMax, amountOut, amountOutMin, amounts } = await getAmounts({ exchange, blockchain, path, pools, tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })); - } catch { return resolve() } + } catch(e) { + console.log(e); + return resolve() + } if([amountIn, amountInMax, amountOut, amountOutMin].every((amount)=>{ return amount == undefined })) { return resolve() } if(exchange.slippage && slippage !== false) { diff --git a/src/exchanges.js b/src/exchanges.js index 5f330b4..d1cb609 100644 --- a/src/exchanges.js +++ b/src/exchanges.js @@ -111,9 +111,11 @@ exchanges.worldchain.forEach((exchange)=>{ exchanges.worldchain[exchange.name] = /*#elif _SOLANA import orca from './exchanges/orca' +import raydium from './exchanges/raydium' const exchanges = [ orca(), + raydium(), ] exchanges.forEach((exchange)=>{ exchanges[exchange.name] = exchange @@ -121,6 +123,7 @@ exchanges.forEach((exchange)=>{ exchanges.solana = [ orca('solana'), + raydium('solana'), ] exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange }) @@ -128,6 +131,7 @@ exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchang import honeyswap from './exchanges/honeyswap' import orca from './exchanges/orca' +import raydium from './exchanges/raydium' import pancakeswap from './exchanges/pancakeswap' import pancakeswap_v3 from './exchanges/pancakeswap_v3' import quickswap from './exchanges/quickswap' @@ -147,6 +151,7 @@ import wxdai from './exchanges/wxdai' const exchanges = [ orca(), + raydium(), uniswap_v3(), pancakeswap_v3(), uniswap_v2(), @@ -193,6 +198,7 @@ exchanges.polygon.forEach((exchange)=>{ exchanges.polygon[exchange.name] = excha exchanges.solana = [ orca('solana'), + raydium('solana'), ] exchanges.solana.forEach((exchange)=>{ exchanges.solana[exchange.name] = exchange }) diff --git a/src/exchanges/raydium.js b/src/exchanges/raydium.js new file mode 100644 index 0000000..cf1ad5e --- /dev/null +++ b/src/exchanges/raydium.js @@ -0,0 +1,48 @@ +import Exchange from '../classes/Exchange' +import Raydium from '../platforms/solana/raydium' + +const exchange = { + + name: 'raydium', + label: 'Raydium', + logo: '', + protocol: 'raydium', + + slippage: true, + + blockchains: ['solana'], + + solana: { + + router_amm: { + address: '675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8', + api: Raydium.AMM_LAYOUT, + }, + + router_cpamm: { + address: 'CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C', + api: Raydium.CPAMM_LAYOUT + }, + + router_clmm: { + address: 'CAMMCzo5YL8w4VFF8KVHrK22GGUsp5VTaW7grrKgrWqK', + api: Raydium.CLMM_LAYOUT + }, + } +} + +export default (scope)=>{ + + return new Exchange( + + Object.assign(exchange, { + scope, + + findPath: (args)=>Raydium.findPath({ ...args, exchange }), + pathExists: (args)=>Raydium.pathExists({ ...args, exchange }), + getAmounts: (args)=>Raydium.getAmounts({ ...args, exchange }), + getPrep: (args)=>{}, + getTransaction: (args)=>Raydium.getTransaction({ ...args, exchange }), + }) + ) +} diff --git a/src/platforms/solana/orca.js b/src/platforms/solana/orca.js deleted file mode 100644 index e67fb93..0000000 --- a/src/platforms/solana/orca.js +++ /dev/null @@ -1,12 +0,0 @@ -import { findPath, pathExists } from './orca/path' -import { getAmounts } from './orca/amounts' -import { getTransaction } from './orca/transaction' -import { WHIRLPOOL_LAYOUT } from './orca/layouts' - -export default { - findPath, - pathExists, - getAmounts, - getTransaction, - WHIRLPOOL_LAYOUT, -} diff --git a/src/platforms/solana/orca/amounts.js b/src/platforms/solana/orca/amounts.js index 81dbb62..1aeff40 100644 --- a/src/platforms/solana/orca/amounts.js +++ b/src/platforms/solana/orca/amounts.js @@ -6,10 +6,14 @@ let getAmountsOut = async ({ path, amountIn, amountInMax }) => { let amounts = [ethers.BigNumber.from(amountIn || amountInMax)] - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax })).price)) + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined })).price)) + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) } if(amounts.length != path.length) { return } @@ -22,10 +26,14 @@ let getAmountsIn = async({ path, amountOut, amountOutMin }) => { path = path.slice().reverse() let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)] - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin })).price)) + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) if (path.length === 3) { - amounts.push(ethers.BigNumber.from((await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined })).price)) + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) } if(amounts.length != path.length) { return } diff --git a/src/platforms/solana/orca/index.js b/src/platforms/solana/orca/index.js new file mode 100644 index 0000000..344ccfa --- /dev/null +++ b/src/platforms/solana/orca/index.js @@ -0,0 +1,12 @@ +import { findPath, pathExists } from './path' +import { getAmounts } from './amounts' +import { getTransaction } from './transaction' +import { WHIRLPOOL_LAYOUT } from './layouts' + +export default { + findPath, + pathExists, + getAmounts, + getTransaction, + WHIRLPOOL_LAYOUT, +} diff --git a/src/platforms/solana/orca/pairs.js b/src/platforms/solana/orca/pairs.js index b9611d2..aff1e55 100644 --- a/src/platforms/solana/orca/pairs.js +++ b/src/platforms/solana/orca/pairs.js @@ -82,7 +82,6 @@ let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, a } else { // amount out bestPair = getLowestPrice(pairs) } - return bestPair } diff --git a/src/platforms/solana/raydium/amm/pairs.js b/src/platforms/solana/raydium/amm/pairs.js new file mode 100644 index 0000000..f7e76a7 --- /dev/null +++ b/src/platforms/solana/raydium/amm/pairs.js @@ -0,0 +1,136 @@ +/*#if _EVM + +/*#elif _SOLANA + +import { request } from '@depay/web3-client-solana' +import { BN } from '@depay/solana-web3.js' + +//#else */ + +import { request } from '@depay/web3-client' +import { BN } from '@depay/solana-web3.js' + +//#endif + +import Blockchains from '@depay/web3-blockchains' +import { AMM_LAYOUT } from '../layouts' + +export const LIQUIDITY_FEES_NUMERATOR = new BN(25) +export const LIQUIDITY_FEES_DENOMINATOR = new BN(10000) + +const BNDivCeil = (bn1, bn2)=> { + const { div, mod } = bn1.divmod(bn2) + + if (mod.gt(new BN(0))) { + return div.add(new BN(1)) + } else { + return div + } +} + +const getPairs = (base, quote)=>{ + return request(`solana://675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8/getProgramAccounts`, { + params: { filters: [ + { dataSize: AMM_LAYOUT.span }, + { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 (Swap) + { memcmp: { offset: 400, bytes: base }}, + { memcmp: { offset: 432, bytes: quote }}, + ]}, + api: AMM_LAYOUT, + cache: 86400, // 24h, + cacheKey: ['raydium', base.toString(), quote.toString()].join('-') + }) +} + +const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + + let accounts = await getPairs(tokenIn, tokenOut) + + if(accounts.length == 0) { + accounts = await getPairs(tokenOut, tokenIn) + } + + const pairs = await Promise.all(accounts.map(async(account)=>{ + + const baseMint = account.data.baseMint.toString() + const quoteMint = account.data.quoteMint.toString() + + const balances = await Promise.all([ + request(`solana://${account.data.baseVault.toString()}/getTokenAccountBalance`, { cache: 5 }), + request(`solana://${account.data.quoteVault.toString()}/getTokenAccountBalance`, { cache: 5 }) + ]) + const baseReserve = (balances[0]?.value?.amount ? new BN(balances[0]?.value?.amount) : new BN(0)).sub(account.data.baseNeedTakePnl) + const quoteReserve = (balances[1]?.value?.amount ? new BN(balances[1]?.value?.amount) : new BN(0)).sub(account.data.quoteNeedTakePnl) + + if(baseMint === Blockchains.solana.wrapped.address) { + if(baseReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (quoteMint === Blockchains.solana.wrapped.address) { + if(quoteReserve.lt(new BN(50000000))) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(baseMint)) { + if((parseFloat(baseReserve.toString()) / 10 ** account.data.baseDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } else if (Blockchains.solana.stables.usd.includes(quoteMint)) { + if((parseFloat(quoteReserve.toString()) / 10 ** account.data.quoteDecimal.toNumber()) < 10000) { + return // to little liquidity + } + } + + const reserves = [baseReserve, quoteReserve] + + if(tokenOut === baseMint) { + reserves.reverse() + } + + const [reserveIn, reserveOut] = reserves + + let price + if(amountOut || amountOutMin) { // compute amountIn + + const amountInRaw = new BN(0) + let amountOutRaw = new BN((amountOut || amountOutMin).toString()) + + if (amountOutRaw.gt(reserveOut)) { + amountOutRaw = reserveOut.sub(new BN(1)) + } + + const denominator = reserveOut.sub(amountOutRaw) + const amountInWithoutFee = reserveIn.mul(amountOutRaw).div(denominator) + + price = amountInWithoutFee + .mul(LIQUIDITY_FEES_DENOMINATOR) + .div(LIQUIDITY_FEES_DENOMINATOR.sub(LIQUIDITY_FEES_NUMERATOR)) + .toString() + + } else { // compute amountOut + + const amountOutRaw = new BN(0) + const amountInRaw = new BN((amountIn || amountInMin).toString()) + const feeRaw = BNDivCeil(amountInRaw.mul(LIQUIDITY_FEES_NUMERATOR), LIQUIDITY_FEES_DENOMINATOR) + + const amountInWithFee = amountInRaw.sub(feeRaw) + const denominator = reserveIn.add(amountInWithFee) + + price = reserveOut.mul(amountInWithFee).div(denominator).toString() + } + + return { + publicKey: account.pubkey.toString(), + baseMint, + quoteMint, + price, + data: account.data + } + + })) + + return pairs.filter(Boolean) +} + +export { + getPairsWithPrice +} diff --git a/src/platforms/solana/raydium/amounts.js b/src/platforms/solana/raydium/amounts.js new file mode 100644 index 0000000..1aeff40 --- /dev/null +++ b/src/platforms/solana/raydium/amounts.js @@ -0,0 +1,101 @@ +import { ethers } from 'ethers' +import { getExchangePath } from './path' +import { getBestPair } from './pairs' + +let getAmountsOut = async ({ path, amountIn, amountInMax }) => { + + let amounts = [ethers.BigNumber.from(amountIn || amountInMax)] + + let bestPair = await getBestPair({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[2], amountIn: amountIn ? amounts[1] : undefined, amountInMax: amountInMax ? amounts[1] : undefined }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) + } + + if(amounts.length != path.length) { return } + + return amounts +} + +let getAmountsIn = async({ path, amountOut, amountOutMin }) => { + + path = path.slice().reverse() + let amounts = [ethers.BigNumber.from(amountOut || amountOutMin)] + + let bestPair = await getBestPair({ tokenIn: path[1], tokenOut: path[0], amountOut, amountOutMin }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) + + if (path.length === 3) { + let bestPair = await getBestPair({ tokenIn: path[2], tokenOut: path[1], amountOut: amountOut ? amounts[1] : undefined, amountOutMin: amountOutMin ? amounts[1] : undefined }) + if(!bestPair){ return } + amounts.push(ethers.BigNumber.from(bestPair.price)) + } + + if(amounts.length != path.length) { return } + + return amounts.slice().reverse() +} + +let getAmounts = async ({ + path, + tokenIn, + tokenOut, + amountOut, + amountIn, + amountInMax, + amountOutMin +}) => { + path = getExchangePath({ path }) + let amounts + if (amountOut) { + amounts = await getAmountsIn({ path, amountOut, tokenIn, tokenOut }) + amountIn = amounts ? amounts[0] : undefined + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn + } + } else if (amountIn) { + amounts = await getAmountsOut({ path, amountIn, tokenIn, tokenOut }) + amountOut = amounts ? amounts[amounts.length-1] : undefined + if (amountOut == undefined || amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut + } + } else if(amountOutMin) { + amounts = await getAmountsIn({ path, amountOutMin, tokenIn, tokenOut }) + amountIn = amounts ? amounts[0] : undefined + if (amountIn == undefined || amountInMax && amountIn.gt(amountInMax)) { + return {} + } else if (amountInMax === undefined) { + amountInMax = amountIn + } + } else if(amountInMax) { + amounts = await getAmountsOut({ path, amountInMax, tokenIn, tokenOut }) + amountOut = amounts ? amounts[amounts.length-1] : undefined + if (amountOut == undefined ||amountOutMin && amountOut.lt(amountOutMin)) { + return {} + } else if (amountOutMin === undefined) { + amountOutMin = amountOut + } + } + return { + amountOut: (amountOut || amountOutMin), + amountIn: (amountIn || amountInMax), + amountInMax: (amountInMax || amountIn), + amountOutMin: (amountOutMin || amountOut), + amounts + } +} + +export { + getAmounts, + getAmountsIn, + getAmountsOut, +} diff --git a/src/platforms/solana/raydium/clmm/pairs.js b/src/platforms/solana/raydium/clmm/pairs.js new file mode 100644 index 0000000..26e573e --- /dev/null +++ b/src/platforms/solana/raydium/clmm/pairs.js @@ -0,0 +1,7 @@ +const getPairsWithPrice = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +} + +export { + getPairsWithPrice +} diff --git a/src/platforms/solana/raydium/cpamm/pairs.js b/src/platforms/solana/raydium/cpamm/pairs.js new file mode 100644 index 0000000..26e573e --- /dev/null +++ b/src/platforms/solana/raydium/cpamm/pairs.js @@ -0,0 +1,7 @@ +const getPairsWithPrice = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + +} + +export { + getPairsWithPrice +} diff --git a/src/platforms/solana/raydium/index.js b/src/platforms/solana/raydium/index.js new file mode 100644 index 0000000..f7766a4 --- /dev/null +++ b/src/platforms/solana/raydium/index.js @@ -0,0 +1,51 @@ +import { findPath, pathExists } from './path' +import { getAmounts } from './amounts' +import { getTransaction } from './transaction' +import { AMM_LAYOUT, CPAMM_LAYOUT, CLMM_LAYOUT, MARKET_LAYOUT } from './layouts' + +export default { + findPath, + pathExists, + getAmounts, + getTransaction, + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, +} + +// // AMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_amm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_amm.api.span }, +// { memcmp: { offset: 0, bytes: [6,0,0] }}, // filters for status 6 +// { memcmp: { offset: 400, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 432, bytes: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_amm.api, +// }) +// console.log(accounts) + + +// // CPAMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_cpamm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_cpamm.api.span }, +// { memcmp: { offset: 168, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 200, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_cpamm.api, +// }) +// console.log(accounts) + + +// // CLMM +// let accounts = await Web3Client.request(`solana://${Web3Exchanges.raydium.solana.router_clmm.address}/getProgramAccounts`, { +// params: { filters: [ +// { dataSize: Web3Exchanges.raydium.solana.router_clmm.api.span }, +// { memcmp: { offset: 73, bytes: "So11111111111111111111111111111111111111112" }}, +// { memcmp: { offset: 105, bytes: "2zMMhcVQEXDtdE6vsFS7S7D5oUodfJHE8vd1gnBouauv" }}, +// ]}, +// api: Web3Exchanges.raydium.solana.router_clmm.api, +// }) +// console.log(accounts) diff --git a/src/platforms/solana/raydium/layouts.js b/src/platforms/solana/raydium/layouts.js new file mode 100644 index 0000000..5b94fac --- /dev/null +++ b/src/platforms/solana/raydium/layouts.js @@ -0,0 +1,176 @@ +import { blob, bool, struct, u8, u16, i32, u32, u64, i128, u128, publicKey, seq } from '@depay/solana-web3.js' + +// OpenBook Market +const MARKET_LAYOUT = struct([ + blob(5), + blob(8), // accountFlagsLayout('accountFlags'), + publicKey('ownAddress'), + u64('vaultSignerNonce'), + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('baseVault'), + u64('baseDepositsTotal'), + u64('baseFeesAccrued'), + publicKey('quoteVault'), + u64('quoteDepositsTotal'), + u64('quoteFeesAccrued'), + u64('quoteDustThreshold'), + publicKey('requestQueue'), + publicKey('eventQueue'), + publicKey('bids'), + publicKey('asks'), + u64('baseLotSize'), + u64('quoteLotSize'), + u64('feeRateBps'), + u64('referrerRebatesAccrued'), + blob(7), +]) + +// AMM 675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8 +const AMM_LAYOUT = struct([ + u64('status'), + u64('nonce'), + u64('maxOrder'), + u64('depth'), + u64('baseDecimal'), + u64('quoteDecimal'), + u64('state'), + u64('resetFlag'), + u64('minSize'), + u64('volMaxCutRatio'), + u64('amountWaveRatio'), + u64('baseLotSize'), + u64('quoteLotSize'), + u64('minPriceMultiplier'), + u64('maxPriceMultiplier'), + u64('systemDecimalValue'), + u64('minSeparateNumerator'), + u64('minSeparateDenominator'), + u64('tradeFeeNumerator'), + u64('tradeFeeDenominator'), + u64('pnlNumerator'), + u64('pnlDenominator'), + u64('swapFeeNumerator'), + u64('swapFeeDenominator'), + u64('baseNeedTakePnl'), + u64('quoteNeedTakePnl'), + u64('quoteTotalPnl'), + u64('baseTotalPnl'), + u64('poolOpenTime'), + u64('punishPcAmount'), + u64('punishCoinAmount'), + u64('orderbookToInitTime'), + u128('swapBaseInAmount'), + u128('swapQuoteOutAmount'), + u64('swapBase2QuoteFee'), + u128('swapQuoteInAmount'), + u128('swapBaseOutAmount'), + u64('swapQuote2BaseFee'), + // amm vault + publicKey('baseVault'), + publicKey('quoteVault'), + // mint + publicKey('baseMint'), + publicKey('quoteMint'), + publicKey('lpMint'), + // market + publicKey('openOrders'), + publicKey('marketId'), + publicKey('marketProgramId'), + publicKey('targetOrders'), + publicKey('withdrawQueue'), + publicKey('lpVault'), + publicKey('owner'), + // true circulating supply without lock up + u64('lpReserve'), + seq(u64(), 3, 'padding'), +]) + +// CP_AMM CPMMoo8L3F4NbTegBCKVNunggL7H1ZpdTHKxQB5qKP1C +const CPAMM_LAYOUT = struct([ + blob(8), + publicKey("configId"), + publicKey("poolCreator"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("mintLp"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("mintProgramA"), + publicKey("mintProgramB"), + publicKey("observationId"), + u8("bump"), + u8("status"), + u8("lpDecimals"), + u8("mintDecimalA"), + u8("mintDecimalB"), + u64("lpAmount"), + u64("protocolFeesMintA"), + u64("protocolFeesMintB"), + u64("fundFeesMintA"), + u64("fundFeesMintB"), + u64("openTime"), + seq(u64(), 32), +]) + +// CLMM + +const RewardInfo = struct([ + u8("rewardState"), + u64("openTime"), + u64("endTime"), + u64("lastUpdateTime"), + u128("emissionsPerSecondX64"), + u64("rewardTotalEmissioned"), + u64("rewardClaimed"), + publicKey("tokenMint"), + publicKey("tokenVault"), + publicKey("creator"), + u128("rewardGrowthGlobalX64"), +]); + +const CLMM_LAYOUT = struct([ + blob(8), + u8("bump"), + publicKey("ammConfig"), + publicKey("creator"), + publicKey("mintA"), + publicKey("mintB"), + publicKey("vaultA"), + publicKey("vaultB"), + publicKey("observationId"), + u8("mintDecimalsA"), + u8("mintDecimalsB"), + u16("tickSpacing"), + u128("liquidity"), + u128("sqrtPriceX64"), + i32("tickCurrent"), + u32(), + u128("feeGrowthGlobalX64A"), + u128("feeGrowthGlobalX64B"), + u64("protocolFeesTokenA"), + u64("protocolFeesTokenB"), + u128("swapInAmountTokenA"), + u128("swapOutAmountTokenB"), + u128("swapInAmountTokenB"), + u128("swapOutAmountTokenA"), + u8("status"), + seq(u8(), 7, ""), + seq(RewardInfo, 3, "rewardInfos"), + seq(u64(), 16, "tickArrayBitmap"), + u64("totalFeesTokenA"), + u64("totalFeesClaimedTokenA"), + u64("totalFeesTokenB"), + u64("totalFeesClaimedTokenB"), + u64("fundFeesTokenA"), + u64("fundFeesTokenB"), + u64("startTime"), + seq(u64(), 15 * 4 - 3, "padding"), +]); + +export { + AMM_LAYOUT, + CPAMM_LAYOUT, + CLMM_LAYOUT, + MARKET_LAYOUT, +} diff --git a/src/platforms/solana/raydium/pairs.js b/src/platforms/solana/raydium/pairs.js new file mode 100644 index 0000000..553b358 --- /dev/null +++ b/src/platforms/solana/raydium/pairs.js @@ -0,0 +1,64 @@ +/*#if _EVM + +/*#elif _SOLANA + +//#else */ + +//#endif + +import { ethers } from 'ethers' +import { getPairsWithPrice as getPairsWithPriceAMM } from './amm/pairs' +import { getPairsWithPrice as getPairsWithPriceCPAMM } from './cpamm/pairs' +import { getPairsWithPrice as getPairsWithPriceCLMM } from './clmm/pairs' + +const getParisWithPriceForAllTypes = ({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + return Promise.all([ + getPairsWithPriceAMM({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPriceCPAMM({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + getPairsWithPriceCLMM({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }), + ]).then((pairsAMM, pairsCPAMM, pairsCLMNN)=>{ + return [ + (pairsAMM || []).filter(Boolean).flat(), + (pairsCPAMM || []).filter(Boolean).flat(), + (pairsCLMNN || []).filter(Boolean).flat() + ].flat() + }) +} + +const getPairsWithPrice = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin })=>{ + try { + return await getParisWithPriceForAllTypes({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + } catch { + return [] + } +} + +let getHighestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).gt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +} + +let getLowestPrice = (pairs)=>{ + return pairs.reduce((bestPricePair, currentPair)=> ethers.BigNumber.from(currentPair.price).lt(ethers.BigNumber.from(bestPricePair.price)) ? currentPair : bestPricePair) +} + +let getBestPair = async({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) => { + const pairs = await getPairsWithPrice({ tokenIn, tokenOut, amountIn, amountInMax, amountOut, amountOutMin }) + + if(!pairs || pairs.length === 0) { return } + + let bestPair + + if(amountIn || amountInMax) { + bestPair = getHighestPrice(pairs) + } else { // amount out + bestPair = getLowestPrice(pairs) + } + + return bestPair +} + +export { + getPairsWithPrice, + getBestPair, +} + diff --git a/src/platforms/solana/raydium/path.js b/src/platforms/solana/raydium/path.js new file mode 100644 index 0000000..c7eeed4 --- /dev/null +++ b/src/platforms/solana/raydium/path.js @@ -0,0 +1,91 @@ +import Blockchains from '@depay/web3-blockchains' +import { getPairsWithPrice } from './pairs' + +const blockchain = Blockchains.solana + +// Replaces 11111111111111111111111111111111 with the wrapped token and implies wrapping. +// +// We keep 11111111111111111111111111111111 internally +// to be able to differentiate between SOL<>Token and WSOL<>Token swaps +// as they are not the same! +// +let getExchangePath = ({ path }) => { + if(!path) { return } + let exchangePath = path.map((token, index) => { + if ( + token === blockchain.currency.address && path[index+1] != blockchain.wrapped.address && + path[index-1] != blockchain.wrapped.address + ) { + return blockchain.wrapped.address + } else { + return token + } + }) + + if(exchangePath[0] == blockchain.currency.address && exchangePath[1] == blockchain.wrapped.address) { + exchangePath.splice(0, 1) + } else if(exchangePath[exchangePath.length-1] == blockchain.currency.address && exchangePath[exchangePath.length-2] == blockchain.wrapped.address) { + exchangePath.splice(exchangePath.length-1, 1) + } + + return exchangePath +} + +let pathExists = async ({ path, amountIn, amountInMax, amountOut, amountOutMin }) => { + if(path.length == 1) { return false } + path = getExchangePath({ path }) + let pairs = [] + if((await getPairsWithPrice({ tokenIn: path[0], tokenOut: path[1], amountIn, amountInMax, amountOut, amountOutMin })).length > 0) { + return true + } else { + return false + } +} + +let findPath = async ({ tokenIn, tokenOut, amountIn, amountOut, amountInMax, amountOutMin }) => { + if( + [tokenIn, tokenOut].includes(blockchain.currency.address) && + [tokenIn, tokenOut].includes(blockchain.wrapped.address) + ) { return { path: undefined, exchangePath: undefined } } + + let path, stablesIn, stablesOut, stable + + if (await pathExists({ path: [tokenIn, tokenOut], amountIn, amountInMax, amountOut, amountOutMin })) { + // direct path + path = [tokenIn, tokenOut] + } else if ( + tokenIn != blockchain.wrapped.address && + tokenIn != blockchain.currency.address && + await pathExists({ path: [tokenIn, blockchain.wrapped.address], amountIn, amountInMax, amountOut, amountOutMin }) && + tokenOut != blockchain.wrapped.address && + tokenOut != blockchain.currency.address && + await pathExists({ path: [tokenOut, blockchain.wrapped.address], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) + ) { + // path via blockchain.wrapped.address + path = [tokenIn, blockchain.wrapped.address, tokenOut] + } else if ( + !blockchain.stables.usd.includes(tokenIn) && + (stablesIn = (await Promise.all(blockchain.stables.usd.map(async(stable)=>await pathExists({ path: [tokenIn, stable], amountIn, amountInMax, amountOut, amountOutMin }) ? stable : undefined))).filter(Boolean)) && + !blockchain.stables.usd.includes(tokenOut) && + (stablesOut = (await Promise.all(blockchain.stables.usd.map(async(stable)=>await pathExists({ path: [tokenOut, stable], amountIn: (amountOut||amountOutMin), amountInMax: (amountOut||amountOutMin), amountOut: (amountIn||amountInMax), amountOutMin: (amountIn||amountInMax) }) ? stable : undefined))).filter(Boolean)) && + (stable = stablesIn.filter((stable)=> stablesOut.includes(stable))[0]) + ) { + // path via TOKEN_IN <> STABLE <> TOKEN_OUT + path = [tokenIn, stable, tokenOut] + } + + // Add blockchain.wrapped.address to route path if things start or end with blockchain.currency.address + // because that actually reflects how things are routed in reality: + if(path?.length && path[0] == blockchain.currency.address) { + path.splice(1, 0, blockchain.wrapped.address) + } else if(path?.length && path[path.length-1] == blockchain.currency.address) { + path.splice(path.length-1, 0, blockchain.wrapped.address) + } + return { path, exchangePath: getExchangePath({ path }) } +} + +export { + findPath, + getExchangePath, + pathExists, +} diff --git a/src/platforms/solana/raydium/transaction.js b/src/platforms/solana/raydium/transaction.js new file mode 100644 index 0000000..53faa59 --- /dev/null +++ b/src/platforms/solana/raydium/transaction.js @@ -0,0 +1,307 @@ +/*#if _EVM + +/*#elif _SOLANA + +import { request, getProvider } from '@depay/web3-client-solana' +import Token from '@depay/web3-tokens-solana' + +//#else */ + +import { request, getProvider } from '@depay/web3-client' +import Token from '@depay/web3-tokens' + +//#endif + +import Blockchains from '@depay/web3-blockchains' +import { Buffer, BN, Transaction, TransactionInstruction, SystemProgram, PublicKey, Keypair, struct, u8, u64, u128, bool } from '@depay/solana-web3.js' +import { getExchangePath } from './path' +import { getBestPair } from './pairs' +import { MARKET_LAYOUT } from './layouts' + +const blockchain = Blockchains.solana + +const getAssociatedAuthority = async(programId)=> { + let [publicKey, nonce] = await PublicKey.findProgramAddress( + // new Uint8Array(Buffer.from('amm authority'.replace('\u00A0', ' '), 'utf-8')) + [Buffer.from([97, 109, 109, 32, 97, 117, 116, 104, 111, 114, 105, 116, 121])], + programId, + ) + return publicKey +} + +const getAssociatedMarketAuthority = async(programId, marketId)=> { + let [publicKey, nonce] = await PublicKey.findProgramAddress( + // Seed is the marketId + [marketId.toBuffer()], + // Program ID for OpenBook/Serum + programId + ) + return publicKey +} + +const getTransaction = async({ + path, + amountIn, + amountInMax, + amountOut, + amountOutMin, + amounts, + amountInInput, + amountOutInput, + amountInMaxInput, + amountOutMinInput, + account +})=>{ + let transaction = { blockchain: 'solana' } + let instructions = [] + + const exchangePath = getExchangePath({ path }) + if(exchangePath.length > 3) { throw 'Raydium can only handle fixed paths with a max length of 3 (2 pools)!' } + const tokenIn = exchangePath[0] + const tokenMiddle = exchangePath.length == 3 ? exchangePath[1] : undefined + const tokenOut = exchangePath[exchangePath.length-1] + + let pairs, amountMiddle + if(exchangePath.length == 2) { + pairs = [await getBestPair({ tokenIn, tokenOut, amountIn: (amountInInput || amountInMaxInput), amountOut: (amountOutInput || amountOutMinInput) })] + } else { + if(amountInInput || amountInMaxInput) { + pairs = [await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountIn: (amountInInput || amountInMaxInput) })] + pairs.push(await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountIn: pairs[0].price })) + } else { // originally amountOut + pairs = [await getBestPair({ tokenIn: tokenMiddle, tokenOut, amountOut: (amountOutInput || amountOutMinInput) })] + pairs.unshift(await getBestPair({ tokenIn, tokenOut: tokenMiddle, amountOut: pairs[0].price })) + } + } + + let startsWrapped = (path[0] === blockchain.currency.address && exchangePath[0] === blockchain.wrapped.address) + let endsUnwrapped = (path[path.length-1] === blockchain.currency.address && exchangePath[exchangePath.length-1] === blockchain.wrapped.address) + let wrappedAccount + const provider = await getProvider('solana') + + if(startsWrapped || endsUnwrapped) { + const rent = await provider.getMinimumBalanceForRentExemption(Token.solana.TOKEN_LAYOUT.span) + const keypair = Keypair.generate() + wrappedAccount = keypair.publicKey.toString() + const lamports = startsWrapped ? new BN(amountIn.toString()).add(new BN(rent)) : new BN(rent) + let createAccountInstruction = SystemProgram.createAccount({ + fromPubkey: new PublicKey(account), + newAccountPubkey: new PublicKey(wrappedAccount), + programId: new PublicKey(Token.solana.TOKEN_PROGRAM), + space: Token.solana.TOKEN_LAYOUT.span, + lamports + }) + createAccountInstruction.signers = [keypair] + instructions.push(createAccountInstruction) + instructions.push( + Token.solana.initializeAccountInstruction({ + account: wrappedAccount, + token: blockchain.wrapped.address, + owner: account + }) + ) + } + + if(pairs.length === 1) { // single hop + const tokenAccountIn = startsWrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenIn })) + const tokenAccountOut = endsUnwrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenOut })) + const pool = pairs[0] + const market = await request(`solana://${pool.data.marketId}/getAccountInfo`, { api: MARKET_LAYOUT }) + + let LAYOUT, data + LAYOUT = struct([ + u8('instruction'), + u64('amountIn'), + u64('minAmountOut') + ]) + data = Buffer.alloc(LAYOUT.span) + LAYOUT.encode( + { + instruction: 9, + amountIn: new BN((amountIn || amountInMax).toString()), + minAmountOut: new BN((amountOutMin || amountOut).toString()), + }, + data, + ) + + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + + // let keys = [ + // // token_program + // { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // // amm + // { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // // amm authority + // { pubkey: await getAssociatedAuthority(new PublicKey(pool.publicKey)), isWritable: false, isSigner: false }, + // // amm openOrders + // { pubkey: new PublicKey(pool.data.openOrders), isWritable: true, isSigner: false }, + // // amm baseVault + // { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // // amm quoteVault + // { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // // openbook marketProgramId + // { pubkey: new PublicKey(pool.data.marketProgramId), isWritable: false, isSigner: false }, + // // openbook marketId + // { pubkey: new PublicKey(pool.data.marketId), isWritable: true, isSigner: false }, + // // openbook marketBids + // { pubkey: market.bids, isWritable: true, isSigner: false }, + // // openbook marketAsks + // { pubkey: market.asks, isWritable: true, isSigner: false }, + // // openbook eventQueue + // { pubkey: market.eventQueue, isWritable: true, isSigner: false }, + // // openbook baseVault + // { pubkey: market.baseVault, isWritable: true, isSigner: false }, + // // openbook quoteVault + // { pubkey: market.quoteVault, isWritable: true, isSigner: false }, + // // openbook marketAuthority + // { pubkey: await getAssociatedMarketAuthority(pool.data.marketProgramId, pool.data.marketId), isWritable: false, isSigner: false }, + // // user tokenAccountIn + // { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // // user tokenAccountOut + // { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // // user owner + // { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + // ] + + let keys = [ + // token_program + { pubkey: new PublicKey(Token.solana.TOKEN_PROGRAM), isWritable: false, isSigner: false }, + // amm + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm authority + { pubkey: new PublicKey('5Q544fKrFoe6tsEbD7S8EmxGTJYAKtTVhAW5Q5pge4j1'), isWritable: false, isSigner: false }, + // amm openOrders + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // amm baseVault + { pubkey: new PublicKey(pool.data.baseVault), isWritable: true, isSigner: false }, + // amm quoteVault + { pubkey: new PublicKey(pool.data.quoteVault), isWritable: true, isSigner: false }, + // openbook marketProgramId + { pubkey: new PublicKey(pool.publicKey), isWritable: false, isSigner: false }, + // openbook marketId + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketBids + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAsks + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook eventQueue + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook baseVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook quoteVault + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // openbook marketAuthority + { pubkey: new PublicKey(pool.publicKey), isWritable: true, isSigner: false }, + // user tokenAccountIn + { pubkey: tokenAccountIn, isWritable: true, isSigner: false }, + // user tokenAccountOut + { pubkey: tokenAccountOut, isWritable: true, isSigner: false }, + // user owner + { pubkey: new PublicKey(account), isWritable: true, isSigner: true } + ] + + console.log('keys', JSON.stringify(keys)) + + instructions.push( + new TransactionInstruction({ + programId: new PublicKey('675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8'), + keys, + data, + }) + ) + } else if (pairs.length === 2) { // two hop + // // amount is NOT the precise part of the swap (otherAmountThreshold is) + // let amountSpecifiedIsInput = !!(amountInInput || amountOutMinInput) + // let amount = amountSpecifiedIsInput ? amountIn : amountOut + // let otherAmountThreshold = amountSpecifiedIsInput ? amountOutMin : amountInMax + // let tokenAccountIn = startsWrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenIn })) + // let tokenMiddle = exchangePath[1] + // let tokenAccountMiddle = new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenMiddle })) + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenMiddle, account: tokenAccountMiddle }) + // let tokenAccountOut = endsUnwrapped ? new PublicKey(wrappedAccount) : new PublicKey(await Token.solana.findProgramAddress({ owner: account, token: tokenOut })) + // if(!endsUnwrapped) { + // await createTokenAccountIfNotExisting({ instructions, owner: account, token: tokenOut, account: tokenAccountOut }) + // } + // instructions.push( + // new TransactionInstruction({ + // programId: new PublicKey('whirLbMiicVdio4qvUfM5KAg6Ct8VwpYzGff3uctyCc'), + // keys: await getTwoHopSwapInstructionKeys({ + // account, + // poolOne: pairs[0].pubkey, + // tickArraysOne: pairs[0].tickArrays, + // tokenAccountOneA: pairs[0].aToB ? tokenAccountIn : tokenAccountMiddle, + // tokenVaultOneA: pairs[0].data.tokenVaultA, + // tokenAccountOneB: pairs[0].aToB ? tokenAccountMiddle : tokenAccountIn, + // tokenVaultOneB: pairs[0].data.tokenVaultB, + // poolTwo: pairs[1].pubkey, + // tickArraysTwo: pairs[1].tickArrays, + // tokenAccountTwoA: pairs[1].aToB ? tokenAccountMiddle : tokenAccountOut, + // tokenVaultTwoA: pairs[1].data.tokenVaultA, + // tokenAccountTwoB: pairs[1].aToB ? tokenAccountOut : tokenAccountMiddle, + // tokenVaultTwoB: pairs[1].data.tokenVaultB, + // }), + // data: getTwoHopSwapInstructionData({ + // amount, + // otherAmountThreshold, + // amountSpecifiedIsInput, + // aToBOne: pairs[0].aToB, + // aToBTwo: pairs[1].aToB, + // sqrtPriceLimitOne: pairs[0].sqrtPriceLimit, + // sqrtPriceLimitTwo: pairs[1].sqrtPriceLimit, + // }), + // }) + // ) + } + + if(startsWrapped || endsUnwrapped) { + instructions.push( + Token.solana.closeAccountInstruction({ + account: wrappedAccount, + owner: account + }) + ) + } + + // await debug(instructions, provider) + + transaction.instructions = instructions + return transaction +} + +const debug = async(instructions, provider)=>{ + console.log('instructions.length', instructions.length) + let data + instructions.forEach((instruction)=>{ + console.log('INSTRUCTION.programId', instruction.programId.toString()) + console.log('INSTRUCTION.keys', instruction.keys) + try { + const LAYOUT = struct([ + u64("anchorDiscriminator"), + u64("amount"), + u64("otherAmountThreshold"), + u128("sqrtPriceLimit"), + bool("amountSpecifiedIsInput"), + bool("aToB"), + ]) + data = LAYOUT.decode(instruction.data) + } catch {} + }) + if(data) { + console.log('INSTRUCTION.data', data) + console.log('amount', data.amount.toString()) + console.log('otherAmountThreshold', data.otherAmountThreshold.toString()) + console.log('sqrtPriceLimit', data.sqrtPriceLimit.toString()) + } + let simulation = new Transaction({ feePayer: new PublicKey('2UgCJaHU5y8NC4uWQcZYeV9a5RyYLF7iKYCybCsdFFD1') }) + instructions.forEach((instruction)=>simulation.add(instruction)) + let result + console.log('SIMULATE') + try{ result = await provider.simulateTransaction(simulation) } catch(e) { console.log('error', e) } + console.log('SIMULATION RESULT', result) +} + +export { + getTransaction, +}