diff --git a/projects/sdk/src/index.ts b/projects/sdk/src/index.ts index 9aafa90a59..633145b929 100644 --- a/projects/sdk/src/index.ts +++ b/projects/sdk/src/index.ts @@ -27,5 +27,11 @@ export type { AdvancedPipeCallStruct as AdvancedPipeStruct } from "src/lib/depot"; +export type { + ZeroExQuoteParams, + ZeroExQuoteResponse, + ZeroExAPIRequestParams +} from "src/lib/matcha/types"; + // Utilities export * as TestUtils from "./utils/TestUtils"; diff --git a/projects/sdk/src/lib/BeanstalkSDK.ts b/projects/sdk/src/lib/BeanstalkSDK.ts index f9b45fedb3..63f44e4bf9 100644 --- a/projects/sdk/src/lib/BeanstalkSDK.ts +++ b/projects/sdk/src/lib/BeanstalkSDK.ts @@ -17,6 +17,7 @@ import defaultSettings from "src/defaultSettings.json"; import { WellsSDK } from "@beanstalk/sdk-wells"; import { ChainId, ChainResolver } from "@beanstalk/sdk-core"; import { Field } from "./field"; +import { ZeroX } from "./matcha"; export type Provider = ethers.providers.JsonRpcProvider; export type Signer = ethers.Signer; @@ -28,6 +29,7 @@ export type BeanstalkConfig = Partial<{ subgraphUrl: string; source: DataSource; DEBUG: boolean; + zeroXApiKey?: string; }>; type Reconfigurable = Pick; @@ -55,6 +57,7 @@ export class BeanstalkSDK { public readonly pools: Pools; public readonly graphql: GraphQLClient; public readonly queries: Queries; + public readonly zeroX: ZeroX; public readonly farm: Farm; public readonly silo: Silo; @@ -83,6 +86,7 @@ export class BeanstalkSDK { this.pools = new Pools(this); this.graphql = new GraphQLClient(this.subgraphUrl); this.queries = getQueries(this.graphql); + this.zeroX = new ZeroX(config?.zeroXApiKey); // // Internal this.events = new EventManager(this); diff --git a/projects/sdk/src/lib/farm/actions/PipelineConvert.ts b/projects/sdk/src/lib/farm/actions/PipelineConvert.ts new file mode 100644 index 0000000000..9d891214ae --- /dev/null +++ b/projects/sdk/src/lib/farm/actions/PipelineConvert.ts @@ -0,0 +1,64 @@ +import { ethers } from "ethers"; +import { BasicPreparedResult, RunContext, StepClass } from "src/classes/Workflow"; +import { BeanstalkSDK } from "src/lib/BeanstalkSDK"; +import { ERC20Token } from "src/classes/Token"; +import { TokenValue } from "@beanstalk/sdk-core"; +import { AdvancedPipeCallStruct } from "src/lib/depot"; + +export class PipelineConvert extends StepClass { + static sdk: BeanstalkSDK; + public name: string = "pipeline-convert"; + + constructor( + private _tokenIn: ERC20Token, + public readonly _stems: ethers.BigNumberish[], + public readonly _amounts: ethers.BigNumberish[], + private _tokenOut: ERC20Token, + private _amountIn: TokenValue, + private _minAmountOut: TokenValue, + public readonly advancedPipeStructs: AdvancedPipeCallStruct[] + ) { + super(); + } + + async run(_amountInStep: ethers.BigNumber, context: RunContext) { + return { + name: this.name, + amountOut: _amountInStep, + prepare: () => { + PipelineConvert.sdk.debug(`[${this.name}.encode()]`, { + tokenIn: this._tokenIn, + amounts: this._amounts, + stems: this._stems, + tokenOut: this._tokenOut, + amountIn: this._amountIn, + minAmountOut: this._minAmountOut, + advancedPipeStructs: this.advancedPipeStructs + }); + return { + target: PipelineConvert.sdk.contracts.beanstalk.address, + callData: PipelineConvert.sdk.contracts.beanstalk.interface.encodeFunctionData( + "pipelineConvert", + [ + this._tokenIn.address, + this._stems, + this._amounts, + this._tokenOut.address, + this.advancedPipeStructs + ] + ) + }; + }, + decode: (data: string) => + PipelineConvert.sdk.contracts.beanstalk.interface.decodeFunctionData( + "pipelineConvert", + data + ), + decodeResult: (result: string) => + PipelineConvert.sdk.contracts.beanstalk.interface.decodeFunctionResult( + "pipelineConvert", + result + ) + }; + } +} diff --git a/projects/sdk/src/lib/farm/actions/index.ts b/projects/sdk/src/lib/farm/actions/index.ts index a754af0078..b014311331 100644 --- a/projects/sdk/src/lib/farm/actions/index.ts +++ b/projects/sdk/src/lib/farm/actions/index.ts @@ -20,6 +20,7 @@ import { UniswapV3Swap } from "./UniswapV3Swap"; import { DevDebug } from "./_DevDebug"; import { LidoWrapSteth } from "./LidoWrapSteth"; import { LidoUnwrapWstETH } from "./LidoUnwrapWstETH"; +import { PipelineConvert } from "./PipelineConvert"; export { // Approvals @@ -44,6 +45,7 @@ export { ClaimWithdrawal, TransferDeposits, TransferDeposit, + PipelineConvert, // Lido LidoWrapSteth, diff --git a/projects/sdk/src/lib/matcha/index.ts b/projects/sdk/src/lib/matcha/index.ts new file mode 100644 index 0000000000..dc0bb44717 --- /dev/null +++ b/projects/sdk/src/lib/matcha/index.ts @@ -0,0 +1,3 @@ +export * from "./types"; + +export { ZeroX } from "./zeroX"; diff --git a/projects/sdk/src/lib/matcha/types.ts b/projects/sdk/src/lib/matcha/types.ts new file mode 100644 index 0000000000..9f2bef486c --- /dev/null +++ b/projects/sdk/src/lib/matcha/types.ts @@ -0,0 +1,284 @@ +export interface ZeroExQuoteParams extends ZeroExAPIRequestParams { + mode: "exactInput" | "exactOutput"; + enabled: boolean; +} + +export interface ZeroExAPIRequestParams { + /** + * The ERC20 token address of the token you want to sell. It is recommended to always use the token address + * instead of token symbols (e.g. ETH ) which may not be recognized by the API. + */ + sellToken: string; + /** + * The ERC20 token address of the token you want to receive. It is recommended to always use the token address + * instead of token symbols (e.g. ETH ) which may not be recognized by the API. + */ + buyToken: string; + /** + * (Optional) The amount of sellToken (in sellToken base units) you want to send. Either sellAmount or buyAmount + * must be present in a request. Specifying sellAmount is the recommended way to interact with + * 0x API as it covers all on-chain sources. + */ + sellAmount?: string; + /** + * (Optional) The amount of buyToken(in buyToken base units) you want to receive. Either sellAmount + * or buyAmount must be present in a request. Note that some on-chain sources do not allow + * specifying buyAmount, when using buyAmount these sources are excluded. + */ + buyAmount?: string; + /** + * (Optional, default is 0.01 for 1%) The maximum acceptable slippage of the buyToken amount if sellAmount + * is provided; The maximum acceptable slippage of the sellAmount amount if buyAmount is provided + * (e.g. 0.03 for 3% slippage allowed). The lowest possible value that can be set for this parameter + * is 0; in other words, no amount of slippage would be allowed. If no value for this optional parameter is + * provided in the API request, the default slippage percentage is 1%. + */ + slippagePercentage?: string; + /** + * (Optional, defaults to ethgasstation "fast") The target gas price (in wei) for the swap transaction. + * If the price is too low to achieve the quote, an error will be returned. + */ + gasPrice?: string; + /** + * (Optional) The address which will fill the quote. While optional, we highly recommend providing this + * parameter if possible so that the API can more accurately estimate the gas required for the swap transaction. + * This helps when validating the entire transaction for success, and catches revert issues. If the validation + * fails, a Revert Error will be returned in the response. The quote should be fillable if this address is provided. + * + * Also, make sure this address has enough token balance. Additionally, including the takerAddress is required + * if you want to integrate RFQ liquidity. + */ + takerAddress?: string; + /** + * (Optional) Liquidity sources (Uniswap, SushiSwap, 0x, Curve, etc) that will not be included in the provided quote. + * See the docs for a full list of sources. + * + * This parameter cannot be combined with includedSources. + */ + excludedSources?: string; + /** + * (Optional) Typically used to filter for RFQ liquidity without any other DEX orders which this is useful + * for testing your RFQ integration. To do so, set it to 0x. + * + * This parameter cannot be combined with excludedSources. + */ + includedSources?: string; + /** + * (Optional) Normally, whenever a takerAddress is provided, the API will validate the quote for the user. + * + * For more details, see "How does takerAddress help with catching issues?" in the docs. + * + * When this parameter is set to true, that validation will be skipped. + * + * Also see Quote Validation in the docs. . + */ + skipValidation?: boolean; + /** + * (Optional) The ETH address that should receive affiliate fees specified with buyTokenPercentageFee. + * Can be used combination with buyTokenPercentageFee to set a commission/trading fee when using the API. + * + * Learn more about how to setup a trading fee/commission fee/transaction fee in the FAQs. + */ + feeRecipient?: string; + /** + * (Optional) The percentage (denoted as a decimal between 0 - 1.0 where 1.0 represents 100%) of + * the buyAmount that should be attributed to feeRecipient as affiliate fees. Note that this requires + * that the feeRecipient parameter is also specified in the request. Learn more about how to setup + * a trading fee/commission fee/transaction fee in the FAQs. + */ + buyTokenPercentageFee?: string; + /** + * (Optional, defaults to 100%) The percentage (between 0 - 1.0) of allowed price impact. + * + * When priceImpactProtectionPercentage is set, estimatedPriceImpact is returned which estimates the change + * in the price of the specified asset that would be caused by the executed swap due to price impact. + * + * If the estimated price impact is above the percentage indicated, an error will be returned. For example, + * if PriceImpactProtectionPercentage=.15 (15%), any quote with a price impact higher than 15% will return an error. + * + * This is an opt-in feature, the default value of 1.0 will disable the feature. When it is set to 1.0 (100%) + * it means that every transaction is allowed to pass. + * + * Note: When we fail to calculate Price Impact we will return null and Price Impact Protection will be disabled + * See affects on estimatedPriceImpact in the Response fields. Read more about price + * impact protection and how to set it up in the docs. + */ + priceImpactProtectionPercentage?: string; + /** + * (Optional) The recipient address of any trade surplus fees. If specified, this address will collect trade surplus + * when applicable. Otherwise, trade surplus will not be collected. + * + * Note: Trade surplus is only sent to this address for sells. It is a no-op for buys. + * Read more about "Can I collect trade surplus?" in the FAQs. + */ + feeRecipientTradeSurplus?: string; + /** + * (Optional) A boolean field. If set to true, the 0x Swap API quote request should sell the entirety of the + * caller's takerToken balance. A sellAmount is still required, even if it is a best guess, because it is + * how a reasonable minimum received amount is determined after slippage. + * + * Note: This parameter is only required for special cases, such as when setting up a multi-step transaction + * or composable operation, where the entire balance is not known ahead of time. Read more about + * "Is there a way to sell assets via Swap API if the exact sellToken amount is not known + * before the transaction is executed?" in the FAQs. + */ + shouldSellEntireBalance?: boolean; +} + +interface ZeroExOrder { + type: number; + source: string; + makerToken: string; + takerToken: string; + makerAmount: string; + takerAmount: string; + fillData: any; + fill: any; +} + +interface ZeroExFee { + feeType: string | "volume"; + feeToken: string; + feeAmount: string; + billingType: string | "on-chain"; +} + +interface ZeroExSource { + name: string; + proportion: string; +} + +/** + * Response type from 0x quote-v1 swap API. + * + * @link https://0x.org/docs/1.0/0x-swap-api/api-references/get-swap-v1-quote + */ +export interface ZeroExQuoteResponse { + /** + * + */ + chainId: number; + /** + * If {buyAmount} was specifed in the request, it provides the price of buyToken in sellToken & vice versa. + * Does not include slippage + */ + price: string; + /** + * Similar to price, but with fees removed from the price calculation. Price as if not fee is charged. + */ + grossPrice: string; + /** + * When priceImpactProtectionPercentage is set, this value returns the estimated change in the price of + * the specified asset that would be caused by the executed swap. + */ + estimatedPriceImpact: string | null; + /** + * The amount of ether (in wei) that should be sent with the transaction. + */ + value: string; + /** + * The gas price (in wei) that should be used to send the transaction. + * The transaction needs to be sent with this gasPrice or lower for the transaction to be successful. + */ + gasPrice: string; + /** + * The estimated gas limit that should be used to send the transaction to guarantee settlement. + * While a computed estimate is returned in all responses, an accurate estimate will only be returned if + * a takerAddress is included in the request. + */ + gas: string; + /** + * The estimate for the amount of gas that will actually be used in the transaction. Always less than gas. + */ + estimatedGas: string; + /** + * The maximum amount of ether (in wei) that will be paid towards the protocol fee, and what is used to compute the value field of the transaction. + * Note, as of ZEIP-91, protocol fees have been removed for all order types. + */ + protocolFee: string; + /** + * The minimum amount of ether (in wei) that will be paid towards the protocol fee during the transaction. + */ + minimumProtocolFee: string; + /** + * The ERC20 token address of the token you want to receive in quote. + */ + buyTokenAddress: string; + /** + * The amount of buyToken (in buyToken units) that would be bought in this swap. + * Certain on-chain sources do not allow specifying buyAmount, when using buyAmount these sources are excluded. + */ + buyAmount: string; + /** + * Similar to buyAmount but with fees removed. This is the buyAmount as if no fee is charged. + */ + grossBuyAmount: string; + /** + * The ERC20 token address of the token you want to sell with quote. + */ + sellTokenAddress: string; + /** + * The amount of sellToken (in sellToken units) that would be sold in this swap. + * Specifying sellAmount is the recommended way to interact with 0xAPI as it covers all on-chain sources. + */ + sellAmount: string; + /** + * Similar to sellAmount but with fees removed. + * This is the sellAmount as if no fee is charged. + * Note: Currently, this will be the same as sellAmount as fees can only be configured to occur on the buyToken. + */ + grossSellAmount: string; + /** + * The percentage distribution of buyAmount or sellAmount split between each liquidity source. + */ + sources: ZeroExSource[]; + /** + * The target contract address for which the user needs to have an allowance in order to be able to complete the swap. + * Typically this is the 0x Exchange Proxy contract address for the specified chain. + * For swaps with "ETH" as sellToken, wrapping "ETH" to "WETH" or unwrapping "WETH" to "ETH" no allowance is needed, + * a null address of 0x0000000000000000000000000000000000000000 is then returned instead. + */ + allowanceTarget: string; + /** + * The rate between ETH and sellToken + */ + sellTokenToEthRate: string; + /** + * The rate between ETH and buyToken + */ + buyTokenToEthRate: string; + /** + * The address of the contract to send call data to. + */ + to: string; + /** + * + */ + from: string; + /** + * The call data + */ + data: string; + /** + * The price which must be met or else the entire transaction will revert. This price is influenced by the slippagePercentage parameter. + * On-chain sources may encounter price movements from quote to settlement. + */ + guaranteedPrice: string; + /** + * The details used to fill orders, used by market makers. If orders is not empty, there will be a type on each order. + * For wrap/unwrap, orders is empty. otherwise, should be populated. + */ + orders: ZeroExOrder[]; + /** + * 0x Swap API fees that would be charged. + */ + fees: Record; + /** + * + */ + decodedUniqueId: string; + /** + * + */ + auxiliaryChainData: any; +} diff --git a/projects/sdk/src/lib/matcha/zeroX.ts b/projects/sdk/src/lib/matcha/zeroX.ts new file mode 100644 index 0000000000..1be78046d9 --- /dev/null +++ b/projects/sdk/src/lib/matcha/zeroX.ts @@ -0,0 +1,93 @@ +import { ZeroExAPIRequestParams, ZeroExQuoteResponse } from "./types"; + +export class ZeroX { + readonly swapV1Endpoint = "https://arbitrum.api.0x.org/swap/v1/quote"; + + constructor(private _apiKey: string = "") {} + + /** + * Exposing here to allow other modules to use their own API key if needed + */ + setApiKey(_apiKey: string) { + this._apiKey = _apiKey; + } + + /** + * fetch the quote from the 0x API + * @notes defaults: + * - slippagePercentage: In human readable form. 0.01 = 1%. Defaults to 0.001 (0.1%) + * - skipValidation: defaults to true + * - shouldSellEntireBalance: defaults to false + */ + async fetchSwapQuote( + args: T, + requestInit?: Omit + ): Promise { + if (!this._apiKey) { + throw new Error("Cannot fetch from 0x without an API key"); + } + + const fetchParams = new URLSearchParams( + this.generateQuoteParams(args) as unknown as Record + ); + + const options = { + ...requestInit, + method: "GET", + headers: new Headers({ + "Content-Type": "application/json", + Accept: "application/json", + "0x-api-key": this._apiKey + }) + }; + + const url = `${this.swapV1Endpoint}?${fetchParams.toString()}`; + + return fetch(url, options).then((r) => r.json()) as Promise; + } + + /** + * Generate the params for the 0x API + * @throws if required params are missing + * + * @returns the params for the 0x API + */ + private generateQuoteParams( + params: T + ): ZeroExAPIRequestParams { + if (!params.buyToken && !params.sellToken) { + throw new Error("buyToken and sellToken and required"); + } + + if (!params.sellAmount && !params.buyAmount) { + throw new Error("sellAmount or buyAmount is required"); + } + + const quoteParams = { + sellToken: params.sellToken, + buyToken: params.buyToken, + sellAmount: params.sellAmount, + buyAmount: params.buyAmount, + slippagePercentage: params.slippagePercentage ?? "0.01", + gasPrice: params.gasPrice, + takerAddress: params.takerAddress, + excludedSources: params.excludedSources, + includedSources: params.includedSources, + skipValidation: params.skipValidation ?? true, // defaults to true b/c most of our swaps go through advFarm / pipeline calls + feeRecipient: params.feeRecipient, + buyTokenPercentageFee: params.buyTokenPercentageFee, + priceImpactProtectionPercentage: params.priceImpactProtectionPercentage, + feeRecipientTradeSurplus: params.feeRecipientTradeSurplus, + shouldSellEntireBalance: params.shouldSellEntireBalance ?? false + }; + + Object.keys(quoteParams).forEach((_key) => { + const key = _key as keyof typeof quoteParams; + if (quoteParams[key] === undefined || quoteParams[key] === null) { + delete quoteParams[key]; + } + }); + + return quoteParams; + } +} diff --git a/projects/sdk/src/lib/silo/GenConvertOperation.ts b/projects/sdk/src/lib/silo/GenConvertOperation.ts deleted file mode 100644 index 0db132ce20..0000000000 --- a/projects/sdk/src/lib/silo/GenConvertOperation.ts +++ /dev/null @@ -1,40 +0,0 @@ -import { BasinWell } from "src/classes/Pool"; -import { BeanstalkSDK } from "../BeanstalkSDK"; -import { ERC20Token } from "src/classes/Token"; -import { AdvancedPipePreparedResult } from "../depot/pipe"; - -export class PipelineConvertOperation { - static sdk: BeanstalkSDK; - - /** - * The whitelisted token to convert from. - */ - readonly inputToken: ERC20Token; - - /** - * The whitelisted token to convert to. - */ - target: ERC20Token; - - advancedPipeCalls: Required[] = []; - - constructor(sdk: BeanstalkSDK, inputToken: ERC20Token) { - PipelineConvertOperation.sdk = sdk; - - this.validateIsWhitelisted(inputToken); - this.inputToken = inputToken; - } - - setTarget(token: ERC20Token) { - this.validateIsWhitelisted(token); - this.target = token; - } - - initialize(token: ERC20Token) {} - - private validateIsWhitelisted(token: ERC20Token) { - if (!PipelineConvertOperation.sdk.tokens.isWhitelisted(token)) { - throw new Error(`GenConvertOperation: Token ${token.symbol} is not whitelisted in the Silo.`); - } - } -} diff --git a/projects/ui/.env.development b/projects/ui/.env.development index 4cdd4e38f5..72b81f1f6d 100644 --- a/projects/ui/.env.development +++ b/projects/ui/.env.development @@ -8,3 +8,8 @@ VITE_ALCHEMY_API_KEY="ds4ljBC_Pq-PaIQ3aHo04t27y2n8qpry" VITE_THEGRAPH_API_KEY="4c0b9364a121c1f2aa96fe61cb73d705" VITE_WALLETCONNECT_PROJECT_ID=2159ea7542f2b547554f8c85eca0cec1 VITE_SNAPSHOT_API_KEY="83b2ba4f5e943503dad56d4afea4a5205ace935d702cb8c0a1151c995b474f59" +VITE_ZERO_X_API_KEY="" +VITE_TENDERLY_ACCOUNT_SLUG="" +VITE_TENDERLY_PROJECT_SLUG="" +VITE_TENDERLY_ACCESS_KEY="" +VITE_TENDERLY_VNET_ID="" \ No newline at end of file diff --git a/projects/ui/.env.production b/projects/ui/.env.production index 346595d3ee..c3ee8bf221 100644 --- a/projects/ui/.env.production +++ b/projects/ui/.env.production @@ -6,3 +6,7 @@ VITE_ALCHEMY_API_KEY="iByabvqm_66b_Bkl9M-wJJGdCTuy19R3" VITE_THEGRAPH_API_KEY="4c0b9364a121c1f2aa96fe61cb73d705" VITE_SNAPSHOT_API_KEY="83b2ba4f5e943503dad56d4afea4a5205ace935d702cb8c0a1151c995b474f59" +VITE_ZERO_X_API_KEY="" +VITE_TENDERLY_ACCOUNT_SLUG="" +VITE_TENDERLY_PROJECT_SLUG="" +VITE_TENDERLY_ACCESS_KEY="" \ No newline at end of file diff --git a/projects/ui/src/components/App/SdkProvider.tsx b/projects/ui/src/components/App/SdkProvider.tsx index 7b409111b9..a63c46d2d2 100644 --- a/projects/ui/src/components/App/SdkProvider.tsx +++ b/projects/ui/src/components/App/SdkProvider.tsx @@ -17,6 +17,10 @@ import rinsableSproutLogo from '~/img/beanstalk/rinsable-sprout-icon.svg'; import beanEthLpLogo from '~/img/tokens/bean-eth-lp-logo.svg'; import beanEthWellLpLogo from '~/img/tokens/bean-eth-well-lp-logo.svg'; import beathWstethWellLPLogo from '~/img/tokens/bean-wsteth-logo.svg'; +import beanUsdcWellLpLogo from '~/img/tokens/bean-usdc-well-lp-logo.svg'; +import beanWbtcWellLpLogo from '~/img/tokens/bean-wbtc-well-lp-logo.svg'; +import beanUsdtWellLpLogo from '~/img/tokens/bean-usdt-well-lp-logo.svg'; +import beanWeethWellLpLogo from '~/img/tokens/bean-weeth-well-lp-logo.svg'; // ERC-20 Token Images import crv3Logo from '~/img/tokens/crv3-logo.png'; @@ -64,10 +68,10 @@ const setTokenMetadatas = (sdk: BeanstalkSDK) => { logo: beathWstethWellLPLogo, }); sdk.tokens.UNRIPE_BEAN_WSTETH.setMetadata({ logo: unripeBeanWstethLogoUrl }); - sdk.tokens.BEAN_WEETH_WELL_LP.setMetadata({ logo: beathWstethWellLPLogo }); // TODO: fix me - sdk.tokens.BEAN_WBTC_WELL_LP.setMetadata({ logo: beathWstethWellLPLogo }); // TODO: fix me - sdk.tokens.BEAN_USDC_WELL_LP.setMetadata({ logo: beathWstethWellLPLogo }); // TODO: fix me - sdk.tokens.BEAN_USDT_WELL_LP.setMetadata({ logo: beathWstethWellLPLogo }); // TODO: fix me + sdk.tokens.BEAN_WEETH_WELL_LP.setMetadata({ logo: beanWeethWellLpLogo }); + sdk.tokens.BEAN_WBTC_WELL_LP.setMetadata({ logo: beanWbtcWellLpLogo }); + sdk.tokens.BEAN_USDC_WELL_LP.setMetadata({ logo: beanUsdcWellLpLogo }); + sdk.tokens.BEAN_USDT_WELL_LP.setMetadata({ logo: beanUsdtWellLpLogo }); // ERC-20 tokens sdk.tokens.BEAN.setMetadata({ logo: beanCircleLogo }); @@ -113,6 +117,7 @@ const useBeanstalkSdkContext = () => { signer: signer ?? undefined, source: datasource, DEBUG: IS_DEVELOPMENT_ENV, + zeroXApiKey: import.meta.env.VITE_ZERO_X_API_KEY, ...(subgraphUrl ? { subgraphUrl } : {}), }); diff --git a/projects/ui/src/components/Silo/Actions/Convert.tsx b/projects/ui/src/components/Silo/Actions/Convert/DefaultConvertForm.tsx similarity index 56% rename from projects/ui/src/components/Silo/Actions/Convert.tsx rename to projects/ui/src/components/Silo/Actions/Convert/DefaultConvertForm.tsx index 4b82de42f6..17eda4aff7 100644 --- a/projects/ui/src/components/Silo/Actions/Convert.tsx +++ b/projects/ui/src/components/Silo/Actions/Convert/DefaultConvertForm.tsx @@ -1,27 +1,11 @@ import React, { useCallback, useEffect, useMemo, useState } from 'react'; import { Box, Stack, Typography, Tooltip, TextField } from '@mui/material'; import HelpOutlineIcon from '@mui/icons-material/HelpOutline'; -import { Form, Formik, FormikHelpers, FormikProps } from 'formik'; +import { Form } from 'formik'; import BigNumber from 'bignumber.js'; -import { - Token, - ERC20Token, - NativeToken, - DataSource, - BeanstalkSDK, - TokenValue, - ConvertDetails, - FarmToMode, - FarmFromMode, -} from '@beanstalk/sdk'; +import { Token, ERC20Token, ConvertDetails } from '@beanstalk/sdk'; import { useSelector } from 'react-redux'; -import { - FormStateNew, - FormTxnsFormState, - SettingInput, - SmartSubmitButton, - TxnSettings, -} from '~/components/Common/Form'; +import { SmartSubmitButton } from '~/components/Common/Form'; import TxnPreview from '~/components/Common/Form/TxnPreview'; import TxnSeparator from '~/components/Common/Form/TxnSeparator'; import PillRow from '~/components/Common/Form/PillRow'; @@ -30,67 +14,31 @@ import { displayBN, displayFullBN, MaxBN, MinBN } from '~/util/Tokens'; import { ZERO_BN } from '~/constants'; import useToggle from '~/hooks/display/useToggle'; import { tokenValueToBN, bnToTokenValue, transform } from '~/util'; -import { FarmerSilo } from '~/state/farmer/silo'; -import useSeason from '~/hooks/beanstalk/useSeason'; -import TransactionToast from '~/components/Common/TxnToast'; import useBDV from '~/hooks/beanstalk/useBDV'; import TokenIcon from '~/components/Common/TokenIcon'; -import { useFetchPools } from '~/state/bean/pools/updater'; import { ActionType } from '~/util/Actions'; -import useFarmerSilo from '~/hooks/farmer/useFarmerSilo'; import { FC } from '~/types'; -import useFormMiddleware from '~/hooks/ledger/useFormMiddleware'; import TokenSelectDialogNew from '~/components/Common/Form/TokenSelectDialogNew'; import TokenQuoteProviderWithParams from '~/components/Common/Form/TokenQuoteProviderWithParams'; -import useSdk from '~/hooks/sdk'; import { QuoteHandlerWithParams } from '~/hooks/ledger/useQuoteWithParams'; -import useAccount from '~/hooks/ledger/useAccount'; import WarningAlert from '~/components/Common/Alert/WarningAlert'; import TokenOutput from '~/components/Common/Form/TokenOutput'; import TxnAccordion from '~/components/Common/TxnAccordion'; import AdditionalTxnsAccordion from '~/components/Common/Form/FormTxn/AdditionalTxnsAccordion'; import useFarmerFormTxnsActions from '~/hooks/farmer/form-txn/useFarmerFormTxnActions'; -import useAsyncMemo from '~/hooks/display/useAsyncMemo'; import AddPlantTxnToggle from '~/components/Common/Form/FormTxn/AddPlantTxnToggle'; -import FormTxnProvider from '~/components/Common/Form/FormTxnProvider'; -import useFormTxnContext from '~/hooks/sdk/useFormTxnContext'; import { FormTxn, ConvertFarmStep } from '~/lib/Txn'; -import usePlantAndDoX from '~/hooks/farmer/form-txn/usePlantAndDoX'; import StatHorizontal from '~/components/Common/StatHorizontal'; import { BeanstalkPalette, FontSize } from '~/components/App/muiTheme'; import { AppState } from '~/state'; +import { ConvertQuoteHandlerParams, BaseConvertFormProps } from './types'; -// ----------------------------------------------------------------------- +interface Props extends BaseConvertFormProps { + conversion: ConvertDetails; + handleQuote: QuoteHandlerWithParams; +} -type ConvertFormValues = FormStateNew & { - settings: { - slippage: number; - }; - maxAmountIn: BigNumber | undefined; - tokenOut: Token | undefined; -} & FormTxnsFormState; - -type ConvertQuoteHandlerParams = { - slippage: number; - isConvertingPlanted: boolean; -}; - -// ----------------------------------------------------------------------- - -const ConvertForm: FC< - FormikProps & { - /** List of tokens that can be converted to. */ - tokenList: (ERC20Token | NativeToken)[]; - /** Farmer's silo balances */ - siloBalances: FarmerSilo['balances']; - handleQuote: QuoteHandlerWithParams; - currentSeason: BigNumber; - /** other */ - sdk: BeanstalkSDK; - conversion: ConvertDetails; - plantAndDoX: ReturnType; - } -> = ({ +export const DefaultConvertForm: FC = ({ tokenList, siloBalances, handleQuote, @@ -554,407 +502,3 @@ const ConvertForm: FC< ); }; - -// ----------------------------------------------------------------------- - -const ConvertPropProvider: FC<{ - fromToken: ERC20Token | NativeToken; -}> = ({ fromToken }) => { - const sdk = useSdk(); - - /// Token List - const [tokenList, initialTokenOut] = useMemo(() => { - // We don't support native token converts - if (fromToken instanceof NativeToken) return [[], undefined]; - const paths = sdk.silo.siloConvert.getConversionPaths(fromToken); - const _tokenList = paths.filter((_token) => !_token.equals(fromToken)); - return [ - _tokenList, // all available tokens to convert to - _tokenList?.[0], // tokenOut is the first available token that isn't the fromToken - ]; - }, [sdk, fromToken]); - - /// Beanstalk - const season = useSeason(); - const [refetchPools] = useFetchPools(); - - /// Farmer - const farmerSilo = useFarmerSilo(); - const farmerSiloBalances = farmerSilo.balances; - const account = useAccount(); - - /// Temporary solution. Remove this when we move the site to use the new sdk types. - const [farmerBalances, refetchFarmerBalances] = useAsyncMemo(async () => { - if (!account) return undefined; - console.debug( - `[Convert] Fetching silo balances for SILO:${fromToken.symbol}` - ); - return sdk.silo.getBalance(fromToken, account, { - source: DataSource.LEDGER, - }); - }, [account, sdk]); - - /// Form - const middleware = useFormMiddleware(); - const { txnBundler, plantAndDoX, refetch } = useFormTxnContext(); - const [conversion, setConversion] = useState({ - actions: [], - amount: TokenValue.ZERO, - bdv: TokenValue.ZERO, - crates: [], - seeds: TokenValue.ZERO, - stalk: TokenValue.ZERO, - }); - - const initialValues: ConvertFormValues = useMemo( - () => ({ - // Settings - settings: { - slippage: 0.05, - }, - // Token Inputs - tokens: [ - { - token: fromToken, - amount: undefined, - quoting: false, - amountOut: undefined, - }, - ], - // Convert data - maxAmountIn: undefined, - // Token Outputs - tokenOut: initialTokenOut, - farmActions: { - preset: fromToken.isLP || fromToken.isUnripe ? 'noPrimary' : 'plant', - primary: undefined, - secondary: undefined, - implied: [FormTxn.MOW], - }, - }), - [fromToken, initialTokenOut] - ); - - /// Handlers - // This handler does not run when _tokenIn = _tokenOut (direct deposit) - const handleQuote = useCallback< - QuoteHandlerWithParams - >( - async (tokenIn, _amountIn, tokenOut, { slippage, isConvertingPlanted }) => { - try { - if (!farmerBalances?.convertibleDeposits) { - throw new Error('No balances found'); - } - const { plantAction } = plantAndDoX; - - const includePlant = !!(isConvertingPlanted && plantAction); - - const result = await ConvertFarmStep._handleConversion( - sdk, - farmerBalances.convertibleDeposits, - tokenIn, - tokenOut, - tokenIn.amount(_amountIn.toString() || '0'), - season.toNumber(), - slippage, - includePlant ? plantAction : undefined - ); - - setConversion(result.conversion); - - return tokenValueToBN(result.minAmountOut); - } catch (e) { - console.debug('[Convert/handleQuote]: FAILED: ', e); - return new BigNumber('0'); - } - }, - [farmerBalances?.convertibleDeposits, sdk, season, plantAndDoX] - ); - - const onSubmit = useCallback( - async ( - values: ConvertFormValues, - formActions: FormikHelpers - ) => { - let txToast; - try { - middleware.before(); - - /// FormData - const slippage = values?.settings?.slippage; - const tokenIn = values.tokens[0].token; - const tokenOut = values.tokenOut; - const _amountIn = values.tokens[0].amount; - - /// Validation - if (!account) throw new Error('Wallet connection required'); - if (!slippage) throw new Error('No slippage value set.'); - if (!tokenOut) throw new Error('Conversion pathway not set'); - if (!farmerBalances) throw new Error('No balances found'); - - txToast = new TransactionToast({ - loading: 'Converting...', - success: 'Convert successful.', - }); - - let txn; - - const { plantAction } = plantAndDoX; - - const amountIn = tokenIn.amount(_amountIn?.toString() || '0'); // amount of from token - const isPlanting = - plantAndDoX && values.farmActions.primary?.includes(FormTxn.PLANT); - - const convertTxn = new ConvertFarmStep( - sdk, - tokenIn, - tokenOut, - season.toNumber(), - farmerBalances.convertibleDeposits - ); - - const { getEncoded, minAmountOut } = await convertTxn.handleConversion( - amountIn, - slippage, - isPlanting ? plantAction : undefined - ); - - convertTxn.build(getEncoded, minAmountOut); - const actionsPerformed = txnBundler.setFarmSteps(values.farmActions); - - if (!isPlanting) { - const { execute } = await txnBundler.bundle( - convertTxn, - amountIn, - slippage, - 1.2 - ); - - txn = await execute(); - } else { - // Create Advanced Farm operation for alt-route Converts - const farm = sdk.farm.createAdvancedFarm('Alternative Convert'); - - // Get Earned Beans data - const stemTips = await sdk.silo.getStemTip(tokenIn); - const earnedBeans = await sdk.silo.getEarnedBeans(account); - const earnedStem = stemTips.toString(); - const earnedAmount = earnedBeans.toBlockchain(); - - // Plant - farm.add(new sdk.farm.actions.Plant()); - - // Withdraw Planted deposit crate - farm.add( - new sdk.farm.actions.WithdrawDeposit( - tokenIn.address, - earnedStem, - earnedAmount, - FarmToMode.INTERNAL - ) - ); - - // Transfer to Well - farm.add( - new sdk.farm.actions.TransferToken( - tokenIn.address, - sdk.pools.BEAN_ETH_WELL.address, - FarmFromMode.INTERNAL, - FarmToMode.EXTERNAL - ) - ); - - // Create Pipeline operation - const pipe = sdk.farm.createAdvancedPipe('pipelineDeposit'); - - // (Pipeline) - Call sync on Well - pipe.add( - new sdk.farm.actions.WellSync( - sdk.pools.BEAN_ETH_WELL, - tokenIn, - sdk.contracts.pipeline.address - ), - { tag: 'amountToDeposit' } - ); - - // (Pipeline) - Approve transfer of sync output - const approveClipboard = { - tag: 'amountToDeposit', - copySlot: 0, - pasteSlot: 1, - }; - pipe.add( - new sdk.farm.actions.ApproveERC20( - sdk.pools.BEAN_ETH_WELL.lpToken, - sdk.contracts.beanstalk.address, - approveClipboard - ) - ); - - // (Pipeline) - Transfer sync output to Beanstalk - const transferClipboard = { - tag: 'amountToDeposit', - copySlot: 0, - pasteSlot: 2, - }; - pipe.add( - new sdk.farm.actions.TransferToken( - sdk.tokens.BEAN_ETH_WELL_LP.address, - account, - FarmFromMode.EXTERNAL, - FarmToMode.INTERNAL, - transferClipboard - ) - ); - - // Add Pipeline operation to the Advanced Pipe operation - farm.add(pipe); - - // Deposit Advanced Pipe output to Silo - farm.add( - new sdk.farm.actions.Deposit( - sdk.tokens.BEAN_ETH_WELL_LP, - FarmFromMode.INTERNAL - ) - ); - - // Convert the other Deposits as usual - if (amountIn.gt(0)) { - const convertData = sdk.silo.siloConvert.calculateConvert( - tokenIn, - tokenOut, - amountIn, - farmerBalances.convertibleDeposits, - season.toNumber() - ); - const amountOut = await sdk.contracts.beanstalk.getAmountOut( - tokenIn.address, - tokenOut.address, - convertData.amount.toBlockchain() - ); - const _minAmountOut = TokenValue.fromBlockchain( - amountOut.toString(), - tokenOut.decimals - ).mul(1 - slippage); - farm.add( - new sdk.farm.actions.Convert( - sdk.tokens.BEAN, - sdk.tokens.BEAN_ETH_WELL_LP, - amountIn, - _minAmountOut, - convertData.crates - ) - ); - } - - // Mow Grown Stalk - const tokensWithStalk: Map = new Map(); - farmerSilo.stalk.grownByToken.forEach((value, token) => { - if (value.gt(0)) { - tokensWithStalk.set(token, value); - } - }); - if (tokensWithStalk.size > 0) { - farm.add(new sdk.farm.actions.Mow(account, tokensWithStalk)); - } - - const gasEstimate = await farm.estimateGas(earnedBeans, { - slippage: slippage, - }); - const adjustedGas = Math.round( - gasEstimate.toNumber() * 1.2 - ).toString(); - txn = await farm.execute( - earnedBeans, - { slippage: slippage }, - { gasLimit: adjustedGas } - ); - } - - txToast.confirming(txn); - - const receipt = await txn.wait(); - - await refetch(actionsPerformed, { farmerSilo: true }, [ - refetchPools, // update prices to account for pool conversion - refetchFarmerBalances, - ]); - - txToast.success(receipt); - - /// Reset the max Amount In - const _maxAmountIn = await ConvertFarmStep.getMaxConvert( - sdk, - tokenIn, - tokenOut - ); - - formActions.resetForm({ - values: { - ...initialValues, - maxAmountIn: tokenValueToBN(_maxAmountIn), - }, - }); - } catch (err) { - console.error(err); - if (txToast) { - txToast.error(err); - } else { - const errorToast = new TransactionToast({}); - errorToast.error(err); - } - formActions.setSubmitting(false); - } - }, - [ - sdk, - season, - account, - txnBundler, - middleware, - plantAndDoX, - initialValues, - farmerBalances, - farmerSilo, - refetch, - refetchPools, - refetchFarmerBalances, - ] - ); - - return ( - - {(formikProps) => ( - <> - - - - - - )} - - ); -}; - -const Convert: FC<{ - fromToken: ERC20Token | NativeToken; -}> = (props) => ( - - - -); - -export default Convert; diff --git a/projects/ui/src/components/Silo/Actions/Convert/PipelineConvertForm.tsx b/projects/ui/src/components/Silo/Actions/Convert/PipelineConvertForm.tsx new file mode 100644 index 0000000000..99be73d6f9 --- /dev/null +++ b/projects/ui/src/components/Silo/Actions/Convert/PipelineConvertForm.tsx @@ -0,0 +1,418 @@ +import React from 'react'; + +import { Form } from 'formik'; +import BigNumber from 'bignumber.js'; +import { useQuery } from '@tanstack/react-query'; +import { + Box, + CircularProgress, + Stack, + Tooltip, + Typography, +} from '@mui/material'; +import HelpOutlineIcon from '@mui/icons-material/HelpOutline'; + +import { BasinWell, Token, TokenSiloBalance, TokenValue } from '@beanstalk/sdk'; + +import { ZERO_BN } from '~/constants'; +import useDebounce from '~/hooks/app/useDebounce'; +import useBDV from '~/hooks/beanstalk/useBDV'; +import useToggle from '~/hooks/display/useToggle'; +import TokenSelectDialog, { + TokenSelectMode, +} from '~/components/Common/Form/TokenSelectDialogNew'; +import { + TokenAdornment, + TokenInputField, + TxnPreview, +} from '~/components/Common/Form'; +import Row from '~/components/Common/Row'; +import PillRow from '~/components/Common/Form/PillRow'; +import TokenIcon from '~/components/Common/TokenIcon'; +import WarningAlert from '~/components/Common/Alert/WarningAlert'; +import TxnAccordion from '~/components/Common/TxnAccordion'; +import StatHorizontal from '~/components/Common/StatHorizontal'; + +import { ActionType, displayFullBN } from '~/util'; +import { PipelineConvertUtil } from '~/lib/PipelineConvert/PipelineConvert'; +import { BaseConvertFormProps } from './types'; + +interface Props extends BaseConvertFormProps { + farmerBalances: TokenSiloBalance | undefined; +} + +interface PipelineConvertFormProps extends Props { + sourceWell: BasinWell; + targetWell: BasinWell; +} + +const baseQueryOptions = { + staleTime: 20_000, // 20 seconds stale time + refetchOnWindowFocus: true, + refetchIntervalInBackground: false, +} as const; + +const PipelineConvertFormInner = ({ + sourceWell, + targetWell, + tokenList, + siloBalances, + farmerBalances: balance, // sdk type + // handleQuote, + sdk, + // Formik + values, + isSubmitting, + setFieldValue, +}: PipelineConvertFormProps) => { + const [tokenSelectOpen, showTokenSelect, hideTokenSelect] = useToggle(); + const getBDV = useBDV(); + + + const sourceToken = sourceWell.lpToken; // LP token of source well + const targetToken = targetWell.lpToken; // LP token of target well + const BEAN = sdk.tokens.BEAN; + + const debouncedAmountIn = useDebounce(values.tokens[0].amount ?? ZERO_BN); // + + const maxConvertableBN = new BigNumber( + (balance?.convertibleAmount || TokenValue.ZERO).toHuman() + ); + + const pickedDeposits = sdk.silo.siloConvert.calculateConvert( + sourceToken, + targetToken, + sourceToken.fromHuman(debouncedAmountIn.toString()), + balance?.convertibleDeposits || [], + 0 + ); + + const sourceIdx = getWellTokenIndexes(sourceWell, BEAN); // token indexes of source well + const targetIdx = getWellTokenIndexes(targetWell, BEAN); // token indexes of target well + + const sellToken = sourceWell.tokens[sourceIdx.nonBeanIndex]; // token we will sell when after removing liquidity in equal parts + const buyToken = targetWell.tokens[targetIdx.nonBeanIndex]; // token we will buy to add liquidity + + const slippage = values.settings.slippage; + + // const amountOut = values.tokens[0].amountOut; // amount of to token + // const maxAmountIn = values.maxAmountIn; + // const canConvert = maxAmountIn?.gt(0) || false; + // const plantCrate = plantAndDoX?.crate?.bn; + + // prettier-ignore + const { data } = useQuery({ + queryKey: ['pipelineConvert', sourceWell.address, targetWell.address, debouncedAmountIn.toString()], + queryFn: async () => { + setFieldValue('tokens.0.quoting', true); + try { + const lpIn = sourceWell.lpToken.fromHuman(debouncedAmountIn.toString()); + const sourceLPAmountOut = await sourceWell.getRemoveLiquidityOutEqual( + lpIn + ); + + console.debug(`[pipelineConvert/removeLiquidity (1)] result:`, { + BEAN: sourceLPAmountOut[sourceIdx.beanIndex].toNumber(), + [`${sellToken.symbol}`]: sourceLPAmountOut[sourceIdx.nonBeanIndex].toNumber(), + }); + + const beanAmountOut = sourceLPAmountOut[sourceIdx.beanIndex]; + const swapAmountIn = sourceLPAmountOut[sourceIdx.nonBeanIndex]; + + const quote = await sdk.zeroX.fetchSwapQuote({ + sellToken: sellToken.address, + buyToken: buyToken.address, + sellAmount: swapAmountIn.blockchainString, + takerAddress: sdk.contracts.pipeline.address, + shouldSellEntireBalance: true, + // 0x requests are formatted such that 0.01 = 1%. Everywhere else in the UI we use 0.01 = 0.01% ?? BS3TODO: VALIDATE ME + slippagePercentage: (slippage * 100).toString(), + }); + + console.debug(`[pipelineConvert/0xQuote (2)] result:`, { quote }); + + const swapAmountOut = buyToken.fromBlockchain(quote?.buyAmount || '0'); + const targetLPAmountOut = await targetWell.getAddLiquidityOut([ + beanAmountOut, + swapAmountOut, + ]); + console.debug(`[pipelineConvert/addLiquidity (3)] result:`, { + amount: targetLPAmountOut.toNumber(), + }); + + setFieldValue('tokens.0.amountOut', new BigNumber(targetLPAmountOut.toHuman())); + return { + amountIn: lpIn, + beanAmountOut, + swapAmountIn, + swapAmountOut, + quote, + targetLPAmountOut, + }; + } catch (e) { + console.debug('[pipelineConvert/query] FAILED: ', e); + throw e; + } finally { + setFieldValue('tokens.0.quoting', false); + } + }, + enabled: maxConvertableBN.gt(0) && debouncedAmountIn?.gt(0), + ...baseQueryOptions, + }); + + const { data: staticCallData } = useQuery({ + queryKey: ['pipelineConvert/callStatic', sourceWell.address, targetWell.address, data?.targetLPAmountOut?.toString()], + queryFn: async () => { + if (!data) return; + try { + const advPipeCalls = PipelineConvertUtil.buildEqual2Equal({ + sdk, + source: { + well: sourceWell, + lpAmountIn: data.amountIn, + beanAmountOut: data.beanAmountOut, + nonBeanAmountOut: data.swapAmountOut, + }, + swap: { + buyToken, + sellToken, + buyAmount: data.swapAmountOut, + quote: data.quote, + }, + target: { + well: targetWell, + amountOut: data.targetLPAmountOut, + }, + slippage, + }); + + const datas = await sdk.contracts.beanstalk.callStatic.pipelineConvert( + sourceToken.address, + pickedDeposits.crates.map((c) => c.stem), + pickedDeposits.crates.map((c) => c.amount.toBigNumber()), + targetToken.address, + advPipeCalls + ).then((result) => ({ + toStem: result.toStem, + fromAmount: result.fromAmount, + toAmount: result.toAmount, + fromBdv: result.fromBdv, + toBdv: result.toBdv, + })); + + console.debug(`[pipelineConvert/callStatic] result:`, datas); + return datas; + } catch (e) { + console.debug('[pipelineConvert/callStatic] FAILED: ', e); + throw e; + } + }, + retry: 2, + enabled: !!data && debouncedAmountIn?.gt(0), + ...baseQueryOptions + }) + + /// When a new output token is selected, reset maxAmountIn. + const handleSelectTokenOut = async (_selectedTokens: Set) => { + const selected = [..._selectedTokens]?.[0]; + + if (!selected || _selectedTokens.size !== 1) { + throw new Error(); + } + + /// only reset if the user clicked a different token + if (targetToken !== selected) { + setFieldValue('tokenOut', selected); + setFieldValue('maxAmountIn', null); + } + }; + + // same as query.isFetching & query.isLoading + const isQuoting = values.tokens?.[0]?.quoting; + + return ( + <> +
+ + + + ), + }} + balance={maxConvertableBN} + quote={ + + + + ~ x instant BDV + {/* ~{displayFullBN(instantBDV, 2, 2)} */} + + + ~ x deposited BDV + {/* ~{displayFullBN(depositBDV, 2, 2)} */} + + + } + placement="top" + > + + + {/* ~{displayFullBN(depositsBDV, 2)} BDV */}x BDV + + + + + {/* {displayQuote(state.amountOut, tokenOut)} */} + {isQuoting && ( + + )} + + } + /> + {maxConvertableBN.gt(0) ? ( + + {targetToken ? : null} + {targetToken?.symbol || 'Select token'} + + ) : null} + + {staticCallData && ( + + values from pipe convert: + + )} + {staticCallData ? (Object.entries(staticCallData).map(([k, v]) => ( + + {k}: {v.toString()} + + ))) : "Failed to load results from static call"} + + {/* You may Lose Grown Stalk warning here */} + + + You may lose Grown Stalk through this transaction. + + + {debouncedAmountIn?.gt(0) && data?.targetLPAmountOut?.gt(0) && ( + + + + + + )} + + + + ); +}; + +export const PipelineConvertForm = ({ values, sdk, ...restProps }: Props) => { + const sourceToken = values.tokens?.[0].token; + const targetToken = values.tokenOut; + + // No need to memoize wells since they their object references don't change + const sourceWell = sourceToken && sdk.pools.getWellByLPToken(sourceToken); + const targetWell = targetToken && sdk.pools.getWellByLPToken(targetToken); + + if (!sourceWell || !targetWell) return null; + + return ( + + ); +}; + +// ------------------------------------------ +// Utils +// ------------------------------------------ + +function getWellTokenIndexes(well: BasinWell | undefined, bean: Token) { + const beanIndex = well?.tokens?.[0].equals(bean) ? 0 : 1; + const nonBeanIndex = beanIndex === 0 ? 1 : 0; + + return { + beanIndex, + nonBeanIndex, + } as const; +} + +// const swapAmountIn = removeOutQuery.data?.[sourceWellNonBeanIndex]; + +// const swapOutQuery = useQuery({ +// queryKey: queryKeys.swapOut(swapTokenIn, swapTokenOut, swapAmountIn), +// queryFn: ({ signal }) => { +// if (!swapTokenIn || !swapTokenOut || !swapAmountIn) return TokenValue.ZERO; +// const controller = new AbortController(); +// signal.addEventListener('abort', () => controller.abort()); + +// const params = sdk.zeroX.fetchQuote({ +// slippagePercentage: values.settings.slippage.toString(), +// buyToken: swapTokenIn.address, +// sellToken: swapTokenOut.address, +// sellAmount: swapAmountIn.blockchainString, +// mode: "" +// }) +// }, +// enabled: !!swapTokenIn && !!swapTokenOut && swapAmountIn?.gt(0), +// initialData: TokenValue.ZERO, +// }); diff --git a/projects/ui/src/components/Silo/Actions/Convert/index.tsx b/projects/ui/src/components/Silo/Actions/Convert/index.tsx new file mode 100644 index 0000000000..5837e4f35a --- /dev/null +++ b/projects/ui/src/components/Silo/Actions/Convert/index.tsx @@ -0,0 +1,940 @@ +import React, { useCallback, useMemo, useState } from 'react'; +import { Formik, FormikHelpers } from 'formik'; +import BigNumber from 'bignumber.js'; +import { + Token, + ERC20Token, + BeanstalkSDK, + TokenValue, + ConvertDetails, + FarmToMode, + FarmFromMode, + TokenSiloBalance, +} from '@beanstalk/sdk'; +import { SettingInput, TxnSettings } from '~/components/Common/Form'; +import { tokenValueToBN } from '~/util'; +import { FarmerSilo } from '~/state/farmer/silo'; +import useSeason from '~/hooks/beanstalk/useSeason'; +import TransactionToast from '~/components/Common/TxnToast'; +import { useFetchPools } from '~/state/bean/pools/updater'; +import useFarmerSilo from '~/hooks/farmer/useFarmerSilo'; +import useFormMiddleware from '~/hooks/ledger/useFormMiddleware'; +import useSdk from '~/hooks/sdk'; +import { QuoteHandlerWithParams } from '~/hooks/ledger/useQuoteWithParams'; +import useAccount from '~/hooks/ledger/useAccount'; +import FormTxnProvider from '~/components/Common/Form/FormTxnProvider'; +import useFormTxnContext from '~/hooks/sdk/useFormTxnContext'; +import { FormTxn, ConvertFarmStep } from '~/lib/Txn'; +import { useWhitelistedTokens } from '~/hooks/beanstalk/useTokens'; +import { useFetchFarmerSilo } from '~/state/farmer/silo/updater'; +import { DefaultConvertForm } from './DefaultConvertForm'; +import { PipelineConvertForm } from './PipelineConvertForm'; +import { + BaseConvertFormProps, + ConvertFormSubmitHandler, + ConvertFormValues, + ConvertProps, + ConvertQuoteHandlerParams, +} from './types'; + +interface Props extends BaseConvertFormProps { + farmerBalances: TokenSiloBalance | undefined; +} + +// ---------- Regular Convert via Beanstalk.convert() ---------- +const DefaultConvertFormWrapper = (props: Props) => { + const { sdk, plantAndDoX, farmerBalances, currentSeason } = props; + + const [conversion, setConversion] = useState({ + actions: [], + amount: TokenValue.ZERO, + bdv: TokenValue.ZERO, + crates: [], + seeds: TokenValue.ZERO, + stalk: TokenValue.ZERO, + }); + + /// Handlers + // This handler does not run when _tokenIn = _tokenOut (direct deposit) + const handleQuote = useCallback< + QuoteHandlerWithParams + >( + async (tokenIn, _amountIn, tokenOut, { slippage, isConvertingPlanted }) => { + try { + if (!farmerBalances?.convertibleDeposits) { + throw new Error('No balances found'); + } + const { plantAction } = plantAndDoX; + + const includePlant = !!(isConvertingPlanted && plantAction); + + const result = await ConvertFarmStep._handleConversion( + sdk, + farmerBalances.convertibleDeposits, + tokenIn, + tokenOut, + tokenIn.amount(_amountIn.toString() || '0'), + currentSeason.toNumber(), + slippage, + includePlant ? plantAction : undefined + ); + + setConversion(result.conversion); + + return tokenValueToBN(result.minAmountOut); + } catch (e) { + console.debug('[Convert/handleQuote]: FAILED: ', e); + return new BigNumber('0'); + } + }, + [farmerBalances?.convertibleDeposits, sdk, currentSeason, plantAndDoX] + ); + + return ( + + ); +}; + +const PipelineConvertFormWrapper = (props: Props) => { + const [conversion, setConversion] = useState({ + // pull this to the parent? + actions: [], + amount: TokenValue.ZERO, + bdv: TokenValue.ZERO, + crates: [], + seeds: TokenValue.ZERO, + stalk: TokenValue.ZERO, + }); + + return ; +}; + +// ---------- Convert Form Router ---------- +/** + * Depending on whether the conversion requires a pipeline convert, + * return the appropriate convert form. + */ +const ConvertFormRouter = (props: Props) => { + const tokenOut = props.values.tokenOut as ERC20Token; + + if (!tokenOut) return null; + + if (isPipelineConvert(props.fromToken, tokenOut)) { + return ; + } + + return ; +}; + +// ---------- Convert Forms Wrapper ---------- +// Acts as a Prop provider for the Convert forms +const ConvertFormsWrapper = ({ fromToken }: ConvertProps) => { + const sdk = useSdk(); + const farmerSilo = useFarmerSilo(); + const season = useSeason(); + + /// Form + const middleware = useFormMiddleware(); + const formTxnContext = useFormTxnContext(); + + /// Token List + const [tokenList, initialTokenOut] = useConvertTokenList(fromToken); + + const initialValues: ConvertFormValues = useMemo( + () => ({ + // Settings + settings: { + slippage: 0.05, + }, + // Token Inputs + tokens: [ + { + token: fromToken, + amount: undefined, + quoting: false, + amountOut: undefined, + }, + ], + // Convert data + maxAmountIn: undefined, + // Token Outputs + tokenOut: initialTokenOut, + farmActions: { + preset: fromToken.isLP || fromToken.isUnripe ? 'noPrimary' : 'plant', + primary: undefined, + secondary: undefined, + implied: [FormTxn.MOW], + }, + }), + [fromToken, initialTokenOut] + ); + + const farmerBalances = farmerSilo.balancesSdk.get(fromToken); + + const defaultSubmitHandler = useDefaultConvertSubmitHandler({ + sdk, + farmerSilo, + middleware, + initialValues, + formTxnContext, + season, + fromToken, + }); + + const submitHandler: ConvertFormSubmitHandler = useCallback( + async (values, formActions) => { + const tokenOut = values.tokenOut; + if (!tokenOut) { + throw new Error('no token out set'); + } + + if (isPipelineConvert(fromToken, tokenOut as ERC20Token)) { + return defaultSubmitHandler(values, formActions); + } + }, + [defaultSubmitHandler, fromToken] + ); + + return ( + + {(formikProps) => ( + <> + + + + + + )} + + ); +}; + +const Convert = (props: ConvertProps) => ( + + + +); + +export default Convert; + +// ----------------------------------------------------------------------- +// Helpers +// ----------------------------------------------------------------------- + +function useDefaultConvertSubmitHandler({ + sdk, + farmerSilo, + middleware, + formTxnContext, + initialValues, + season, + fromToken, +}: { + sdk: BeanstalkSDK; + farmerSilo: FarmerSilo; + middleware: ReturnType; + formTxnContext: ReturnType; + initialValues: ConvertFormValues; + season: BigNumber; + fromToken: ERC20Token; +}) { + const { txnBundler, plantAndDoX, refetch } = formTxnContext; + const account = useAccount(); + const [refetchFarmerSilo] = useFetchFarmerSilo(); + const [refetchPools] = useFetchPools(); + + return useCallback( + async ( + values: ConvertFormValues, + formActions: FormikHelpers + ) => { + const farmerBalances = farmerSilo.balancesSdk.get(fromToken); + let txToast; + try { + middleware.before(); + + /// FormData + const slippage = values?.settings?.slippage; + const tokenIn = values.tokens[0].token; + const tokenOut = values.tokenOut; + const _amountIn = values.tokens[0].amount; + + /// Validation + if (!account) throw new Error('Wallet connection required'); + if (!slippage) throw new Error('No slippage value set.'); + if (!tokenOut) throw new Error('Conversion pathway not set'); + if (!farmerBalances) throw new Error('No balances found'); + + txToast = new TransactionToast({ + loading: 'Converting...', + success: 'Convert successful.', + }); + + let txn; + + const { plantAction } = plantAndDoX; + + const amountIn = tokenIn.amount(_amountIn?.toString() || '0'); // amount of from token + const isPlanting = + plantAndDoX && values.farmActions.primary?.includes(FormTxn.PLANT); + + const convertTxn = new ConvertFarmStep( + sdk, + tokenIn, + tokenOut, + season.toNumber(), + farmerBalances.convertibleDeposits + ); + + const { getEncoded, minAmountOut } = await convertTxn.handleConversion( + amountIn, + slippage, + isPlanting ? plantAction : undefined + ); + + convertTxn.build(getEncoded, minAmountOut); + const actionsPerformed = txnBundler.setFarmSteps(values.farmActions); + + if (!isPlanting) { + const { execute } = await txnBundler.bundle( + convertTxn, + amountIn, + slippage, + 1.2 + ); + + txn = await execute(); + } else { + // Create Advanced Farm operation for alt-route Converts + const farm = sdk.farm.createAdvancedFarm('Alternative Convert'); + + // Get Earned Beans data + const stemTips = await sdk.silo.getStemTip(tokenIn); + const earnedBeans = await sdk.silo.getEarnedBeans(account); + const earnedStem = stemTips.toString(); + const earnedAmount = earnedBeans.toBlockchain(); + + // Plant + farm.add(new sdk.farm.actions.Plant()); + + // Withdraw Planted deposit crate + farm.add( + new sdk.farm.actions.WithdrawDeposit( + tokenIn.address, + earnedStem, + earnedAmount, + FarmToMode.INTERNAL + ) + ); + + // Transfer to Well + farm.add( + new sdk.farm.actions.TransferToken( + tokenIn.address, + sdk.pools.BEAN_ETH_WELL.address, + FarmFromMode.INTERNAL, + FarmToMode.EXTERNAL + ) + ); + + // Create Pipeline operation + const pipe = sdk.farm.createAdvancedPipe('pipelineDeposit'); + + // (Pipeline) - Call sync on Well + pipe.add( + new sdk.farm.actions.WellSync( + sdk.pools.BEAN_ETH_WELL, + tokenIn, + sdk.contracts.pipeline.address + ), + { tag: 'amountToDeposit' } + ); + + // (Pipeline) - Approve transfer of sync output + const approveClipboard = { + tag: 'amountToDeposit', + copySlot: 0, + pasteSlot: 1, + }; + pipe.add( + new sdk.farm.actions.ApproveERC20( + sdk.pools.BEAN_ETH_WELL.lpToken, + sdk.contracts.beanstalk.address, + approveClipboard + ) + ); + + // (Pipeline) - Transfer sync output to Beanstalk + const transferClipboard = { + tag: 'amountToDeposit', + copySlot: 0, + pasteSlot: 2, + }; + pipe.add( + new sdk.farm.actions.TransferToken( + sdk.tokens.BEAN_ETH_WELL_LP.address, + account, + FarmFromMode.EXTERNAL, + FarmToMode.INTERNAL, + transferClipboard + ) + ); + + // Add Pipeline operation to the Advanced Pipe operation + farm.add(pipe); + + // Deposit Advanced Pipe output to Silo + farm.add( + new sdk.farm.actions.Deposit( + sdk.tokens.BEAN_ETH_WELL_LP, + FarmFromMode.INTERNAL + ) + ); + + // Convert the other Deposits as usual + if (amountIn.gt(0)) { + const convertData = sdk.silo.siloConvert.calculateConvert( + tokenIn, + tokenOut, + amountIn, + farmerBalances.convertibleDeposits, + season.toNumber() + ); + const amountOut = await sdk.contracts.beanstalk.getAmountOut( + tokenIn.address, + tokenOut.address, + convertData.amount.toBlockchain() + ); + const _minAmountOut = TokenValue.fromBlockchain( + amountOut.toString(), + tokenOut.decimals + ).mul(1 - slippage); + farm.add( + new sdk.farm.actions.Convert( + sdk.tokens.BEAN, + sdk.tokens.BEAN_ETH_WELL_LP, + amountIn, + _minAmountOut, + convertData.crates + ) + ); + } + + // Mow Grown Stalk + const tokensWithStalk: Map = new Map(); + farmerSilo.stalk.grownByToken.forEach((value, token) => { + if (value.gt(0)) { + tokensWithStalk.set(token, value); + } + }); + if (tokensWithStalk.size > 0) { + farm.add(new sdk.farm.actions.Mow(account, tokensWithStalk)); + } + + const gasEstimate = await farm.estimateGas(earnedBeans, { + slippage: slippage, + }); + const adjustedGas = Math.round( + gasEstimate.toNumber() * 1.2 + ).toString(); + txn = await farm.execute( + earnedBeans, + { slippage: slippage }, + { gasLimit: adjustedGas } + ); + } + + txToast.confirming(txn); + + const receipt = await txn.wait(); + + await refetch(actionsPerformed, { farmerSilo: true }, [ + refetchPools, // update prices to account for pool conversion + refetchFarmerSilo, + ]); + + txToast.success(receipt); + + /// Reset the max Amount In + const _maxAmountIn = await ConvertFarmStep.getMaxConvert( + sdk, + tokenIn, + tokenOut + ); + + formActions.resetForm({ + values: { + ...initialValues, + maxAmountIn: tokenValueToBN(_maxAmountIn), + }, + }); + } catch (err) { + console.error(err); + if (txToast) { + txToast.error(err); + } else { + const errorToast = new TransactionToast({}); + errorToast.error(err); + } + formActions.setSubmitting(false); + } + }, + [ + fromToken, + sdk, + season, + account, + txnBundler, + middleware, + plantAndDoX, + initialValues, + farmerSilo, + refetch, + refetchPools, + refetchFarmerSilo, + ] + ); +} + +function isPipelineConvert( + fromToken: ERC20Token, + toToken: ERC20Token | undefined +) { + if (!toToken) return false; + if (fromToken.isLP && toToken.isLP) { + // Make sure it isn't a lambda convert + return !fromToken.equals(toToken); + } + return false; +} + +function useConvertTokenList( + fromToken: ERC20Token +): [tokenList: ERC20Token[], initialTokenOut: ERC20Token] { + const { whitelist, tokenMap: whitelistLookup } = useWhitelistedTokens(); + const sdk = useSdk(); + return useMemo(() => { + const pathSet = new Set( + sdk.silo.siloConvert.getConversionPaths(fromToken).map((t) => t.address) + ); + + if (!fromToken.isUnripe) { + whitelist.forEach((toToken) => { + !toToken.isUnripe && pathSet.add(toToken.address); + }); + } + + const list = Array.from(pathSet).map((address) => whitelistLookup[address]); + return [ + list, // all available tokens to convert to + list?.[0], // tokenOut is the first available token that isn't the fromToken + ]; + }, [fromToken, sdk, whitelist, whitelistLookup]); +} + +// const ConvertPropProvider: FC<{ +// fromToken: ERC20Token; +// }> = ({ fromToken }) => { +// const sdk = useSdk(); + +// /// Token List +// const [tokenList, initialTokenOut] = useMemo(() => { +// const paths = sdk.silo.siloConvert.getConversionPaths(fromToken); +// const _tokenList = paths.filter((_token) => !_token.equals(fromToken)); +// return [ +// _tokenList, // all available tokens to convert to +// _tokenList?.[0], // tokenOut is the first available token that isn't the fromToken +// ]; +// }, [sdk, fromToken]); + +// /// Beanstalk +// const season = useSeason(); +// const [refetchPools] = useFetchPools(); + +// /// Farmer +// const farmerSilo = useFarmerSilo(); +// const farmerSiloBalances = farmerSilo.balances; +// const account = useAccount(); + +// /// Temporary solution. Remove this when we move the site to use the new sdk types. +// const [farmerBalances, refetchFarmerBalances] = useAsyncMemo(async () => { +// if (!account) return undefined; +// console.debug( +// `[Convert] Fetching silo balances for SILO:${fromToken.symbol}` +// ); +// return sdk.silo.getBalance(fromToken, account, { +// source: DataSource.LEDGER, +// }); +// }, [account, sdk]); + +// /// Form +// const middleware = useFormMiddleware(); +// const { txnBundler, plantAndDoX, refetch } = useFormTxnContext(); +// const [conversion, setConversion] = useState({ +// actions: [], +// amount: TokenValue.ZERO, +// bdv: TokenValue.ZERO, +// crates: [], +// seeds: TokenValue.ZERO, +// stalk: TokenValue.ZERO, +// }); + +// const initialValues: ConvertFormValues = useMemo( +// () => ({ +// // Settings +// settings: { +// slippage: 0.05, +// }, +// // Token Inputs +// tokens: [ +// { +// token: fromToken, +// amount: undefined, +// quoting: false, +// amountOut: undefined, +// }, +// ], +// // Convert data +// maxAmountIn: undefined, +// // Token Outputs +// tokenOut: initialTokenOut, +// farmActions: { +// preset: fromToken.isLP || fromToken.isUnripe ? 'noPrimary' : 'plant', +// primary: undefined, +// secondary: undefined, +// implied: [FormTxn.MOW], +// }, +// }), +// [fromToken, initialTokenOut] +// ); + +// /// Handlers +// // This handler does not run when _tokenIn = _tokenOut (direct deposit) +// const handleQuote = useCallback< +// QuoteHandlerWithParams +// >( +// async (tokenIn, _amountIn, tokenOut, { slippage, isConvertingPlanted }) => { +// try { +// if (!farmerBalances?.convertibleDeposits) { +// throw new Error('No balances found'); +// } +// const { plantAction } = plantAndDoX; + +// const includePlant = !!(isConvertingPlanted && plantAction); + +// const result = await ConvertFarmStep._handleConversion( +// sdk, +// farmerBalances.convertibleDeposits, +// tokenIn, +// tokenOut, +// tokenIn.amount(_amountIn.toString() || '0'), +// season.toNumber(), +// slippage, +// includePlant ? plantAction : undefined +// ); + +// setConversion(result.conversion); + +// return tokenValueToBN(result.minAmountOut); +// } catch (e) { +// console.debug('[Convert/handleQuote]: FAILED: ', e); +// return new BigNumber('0'); +// } +// }, +// [farmerBalances?.convertibleDeposits, sdk, season, plantAndDoX] +// ); + +// const onSubmit = useCallback( +// async ( +// values: ConvertFormValues, +// formActions: FormikHelpers +// ) => { +// let txToast; +// try { +// middleware.before(); + +// /// FormData +// const slippage = values?.settings?.slippage; +// const tokenIn = values.tokens[0].token; +// const tokenOut = values.tokenOut; +// const _amountIn = values.tokens[0].amount; + +// /// Validation +// if (!account) throw new Error('Wallet connection required'); +// if (!slippage) throw new Error('No slippage value set.'); +// if (!tokenOut) throw new Error('Conversion pathway not set'); +// if (!farmerBalances) throw new Error('No balances found'); + +// txToast = new TransactionToast({ +// loading: 'Converting...', +// success: 'Convert successful.', +// }); + +// let txn; + +// const { plantAction } = plantAndDoX; + +// const amountIn = tokenIn.amount(_amountIn?.toString() || '0'); // amount of from token +// const isPlanting = +// plantAndDoX && values.farmActions.primary?.includes(FormTxn.PLANT); + +// const convertTxn = new ConvertFarmStep( +// sdk, +// tokenIn, +// tokenOut, +// season.toNumber(), +// farmerBalances.convertibleDeposits +// ); + +// const { getEncoded, minAmountOut } = await convertTxn.handleConversion( +// amountIn, +// slippage, +// isPlanting ? plantAction : undefined +// ); + +// convertTxn.build(getEncoded, minAmountOut); +// const actionsPerformed = txnBundler.setFarmSteps(values.farmActions); + +// if (!isPlanting) { +// const { execute } = await txnBundler.bundle( +// convertTxn, +// amountIn, +// slippage, +// 1.2 +// ); + +// txn = await execute(); +// } else { +// // Create Advanced Farm operation for alt-route Converts +// const farm = sdk.farm.createAdvancedFarm('Alternative Convert'); + +// // Get Earned Beans data +// const stemTips = await sdk.silo.getStemTip(tokenIn); +// const earnedBeans = await sdk.silo.getEarnedBeans(account); +// const earnedStem = stemTips.toString(); +// const earnedAmount = earnedBeans.toBlockchain(); + +// // Plant +// farm.add(new sdk.farm.actions.Plant()); + +// // Withdraw Planted deposit crate +// farm.add( +// new sdk.farm.actions.WithdrawDeposit( +// tokenIn.address, +// earnedStem, +// earnedAmount, +// FarmToMode.INTERNAL +// ) +// ); + +// // Transfer to Well +// farm.add( +// new sdk.farm.actions.TransferToken( +// tokenIn.address, +// sdk.pools.BEAN_ETH_WELL.address, +// FarmFromMode.INTERNAL, +// FarmToMode.EXTERNAL +// ) +// ); + +// // Create Pipeline operation +// const pipe = sdk.farm.createAdvancedPipe('pipelineDeposit'); + +// // (Pipeline) - Call sync on Well +// pipe.add( +// new sdk.farm.actions.WellSync( +// sdk.pools.BEAN_ETH_WELL, +// tokenIn, +// sdk.contracts.pipeline.address +// ), +// { tag: 'amountToDeposit' } +// ); + +// // (Pipeline) - Approve transfer of sync output +// const approveClipboard = { +// tag: 'amountToDeposit', +// copySlot: 0, +// pasteSlot: 1, +// }; +// pipe.add( +// new sdk.farm.actions.ApproveERC20( +// sdk.pools.BEAN_ETH_WELL.lpToken, +// sdk.contracts.beanstalk.address, +// approveClipboard +// ) +// ); + +// // (Pipeline) - Transfer sync output to Beanstalk +// const transferClipboard = { +// tag: 'amountToDeposit', +// copySlot: 0, +// pasteSlot: 2, +// }; +// pipe.add( +// new sdk.farm.actions.TransferToken( +// sdk.tokens.BEAN_ETH_WELL_LP.address, +// account, +// FarmFromMode.EXTERNAL, +// FarmToMode.INTERNAL, +// transferClipboard +// ) +// ); + +// // Add Pipeline operation to the Advanced Pipe operation +// farm.add(pipe); + +// // Deposit Advanced Pipe output to Silo +// farm.add( +// new sdk.farm.actions.Deposit( +// sdk.tokens.BEAN_ETH_WELL_LP, +// FarmFromMode.INTERNAL +// ) +// ); + +// // Convert the other Deposits as usual +// if (amountIn.gt(0)) { +// const convertData = sdk.silo.siloConvert.calculateConvert( +// tokenIn, +// tokenOut, +// amountIn, +// farmerBalances.convertibleDeposits, +// season.toNumber() +// ); +// const amountOut = await sdk.contracts.beanstalk.getAmountOut( +// tokenIn.address, +// tokenOut.address, +// convertData.amount.toBlockchain() +// ); +// const _minAmountOut = TokenValue.fromBlockchain( +// amountOut.toString(), +// tokenOut.decimals +// ).mul(1 - slippage); +// farm.add( +// new sdk.farm.actions.Convert( +// sdk.tokens.BEAN, +// sdk.tokens.BEAN_ETH_WELL_LP, +// amountIn, +// _minAmountOut, +// convertData.crates +// ) +// ); +// } + +// // Mow Grown Stalk +// const tokensWithStalk: Map = new Map(); +// farmerSilo.stalk.grownByToken.forEach((value, token) => { +// if (value.gt(0)) { +// tokensWithStalk.set(token, value); +// } +// }); +// if (tokensWithStalk.size > 0) { +// farm.add(new sdk.farm.actions.Mow(account, tokensWithStalk)); +// } + +// const gasEstimate = await farm.estimateGas(earnedBeans, { +// slippage: slippage, +// }); +// const adjustedGas = Math.round( +// gasEstimate.toNumber() * 1.2 +// ).toString(); +// txn = await farm.execute( +// earnedBeans, +// { slippage: slippage }, +// { gasLimit: adjustedGas } +// ); +// } + +// txToast.confirming(txn); + +// const receipt = await txn.wait(); + +// await refetch(actionsPerformed, { farmerSilo: true }, [ +// refetchPools, // update prices to account for pool conversion +// refetchFarmerBalances, +// ]); + +// txToast.success(receipt); + +// /// Reset the max Amount In +// const _maxAmountIn = await ConvertFarmStep.getMaxConvert( +// sdk, +// tokenIn, +// tokenOut +// ); + +// formActions.resetForm({ +// values: { +// ...initialValues, +// maxAmountIn: tokenValueToBN(_maxAmountIn), +// }, +// }); +// } catch (err) { +// console.error(err); +// if (txToast) { +// txToast.error(err); +// } else { +// const errorToast = new TransactionToast({}); +// errorToast.error(err); +// } +// formActions.setSubmitting(false); +// } +// }, +// [ +// sdk, +// season, +// account, +// txnBundler, +// middleware, +// plantAndDoX, +// initialValues, +// farmerBalances, +// farmerSilo, +// refetch, +// refetchPools, +// refetchFarmerBalances, +// ] +// ); + +// return ( +// +// {(formikProps) => ( +// <> +// +// +// +// +// +// )} +// +// ); +// }; diff --git a/projects/ui/src/components/Silo/Actions/Convert/types.ts b/projects/ui/src/components/Silo/Actions/Convert/types.ts new file mode 100644 index 0000000000..c58779dd57 --- /dev/null +++ b/projects/ui/src/components/Silo/Actions/Convert/types.ts @@ -0,0 +1,47 @@ +import { BeanstalkSDK, ERC20Token, NativeToken, Token } from '@beanstalk/sdk'; +import BigNumber from 'bignumber.js'; +import { FormikHelpers, FormikProps } from 'formik'; +import { FormStateNew, FormTxnsFormState } from '~/components/Common/Form'; +import usePlantAndDoX from '~/hooks/farmer/form-txn/usePlantAndDoX'; +import { FarmerSilo } from '~/state/farmer/silo'; + +// ---------- FORMIK ---------- +export type ConvertFormValues = FormStateNew & { + settings: { + slippage: number; + }; + maxAmountIn: BigNumber | undefined; + tokenOut: Token | undefined; +} & FormTxnsFormState; + +export type ConvertFormSubmitHandler = ( + values: ConvertFormValues, + formActions: FormikHelpers +) => Promise; + +export type ConvertQuoteHandlerParams = { + slippage: number; + isConvertingPlanted: boolean; +}; + +// ---------- COMPONENT ---------- + +type BaseConvertFormikProps = FormikProps; + +// Base Props +export interface ConvertProps { + fromToken: ERC20Token; +} + +export interface BaseConvertFormProps + extends ConvertProps, + BaseConvertFormikProps { + /** List of tokens that can be converted to. */ + tokenList: (ERC20Token | NativeToken)[]; + /** Farmer's silo balances */ + siloBalances: FarmerSilo['balances']; + currentSeason: BigNumber; + /** other */ + sdk: BeanstalkSDK; + plantAndDoX: ReturnType; +} diff --git a/projects/ui/src/constants/tokens.ts b/projects/ui/src/constants/tokens.ts index 2a635650cb..7b21aec4e4 100644 --- a/projects/ui/src/constants/tokens.ts +++ b/projects/ui/src/constants/tokens.ts @@ -17,6 +17,10 @@ import rinsableSproutLogo from '~/img/beanstalk/rinsable-sprout-icon.svg'; import beanEthLpLogoUrl from '~/img/tokens/bean-eth-lp-logo.svg'; import beanEthWellLpLogoUrl from '~/img/tokens/bean-eth-well-lp-logo.svg'; import beanLusdLogoUrl from '~/img/tokens/bean-lusd-logo.svg'; +import beanUsdcWellLpLogo from '~/img/tokens/bean-usdc-well-lp-logo.svg'; +import beanWbtcWellLpLogo from '~/img/tokens/bean-wbtc-well-lp-logo.svg'; +import beanUsdtWellLpLogo from '~/img/tokens/bean-usdt-well-lp-logo.svg'; +import beanWeethWellLpLogo from '~/img/tokens/bean-weeth-well-lp-logo.svg'; // ERC-20 Token Images import wstethLogo from '~/img/tokens/wsteth-logo.svg'; @@ -301,7 +305,7 @@ export const BEAN_WEETH_WELL_LP = makeChainToken( name: 'BEAN:weETH LP', symbol: 'BEANweETH', isLP: true, - logo: beanWstethLogo, // TODO: replace with bean:weeth logo + logo: beanWeethWellLpLogo, isUnripe: false, displayDecimals: 2, }, @@ -316,7 +320,7 @@ export const BEAN_WBTC_WELL_LP = makeChainToken( symbol: 'BEANWBTC', isLP: true, isUnripe: false, - logo: beanWstethLogo, // TODO: replace with bean:weeth logo + logo: beanWbtcWellLpLogo, displayDecimals: 2, }, { ...defaultRewards } @@ -330,7 +334,7 @@ export const BEAN_USDC_WELL_LP = makeChainToken( symbol: 'BEANUSDC', isLP: true, isUnripe: false, - logo: beanWstethLogo, // TODO: replace with bean:weeth logo + logo: beanUsdcWellLpLogo, displayDecimals: 2, }, { ...defaultRewards } @@ -344,7 +348,7 @@ export const BEAN_USDT_WELL_LP = makeChainToken( symbol: 'BEANUSDT', isLP: true, isUnripe: false, - logo: beanWstethLogo, // TODO: replace with bean:weeth logo + logo: beanUsdtWellLpLogo, displayDecimals: 2, }, { ...defaultRewards } diff --git a/projects/ui/src/env.d.ts b/projects/ui/src/env.d.ts index 9e5d4b6096..3180c07ec0 100644 --- a/projects/ui/src/env.d.ts +++ b/projects/ui/src/env.d.ts @@ -21,6 +21,31 @@ interface ImportMetaEnv { * If set, don't add CSP meta tag */ readonly DISABLE_CSP?: any; + + /** + * API key for used for ZeroX Swap API + */ + readonly VITE_ZERO_X_API_KEY: string; + + /** + * + */ + readonly VITE_TENDERLY_ACCOUNT_SLUG: string; + + /** + * + */ + readonly VITE_TENDERLY_PROJECT_SLUG: string; + + /** + * + */ + readonly VITE_TENDERLY_ACCESS_KEY: string; + + /** + * Tenderly Virtual Network ID for testnet + */ + readonly VITE_TENDERLY_VNET_ID?: string; } interface ImportMeta { diff --git a/projects/ui/src/graph/graphql.schema.json b/projects/ui/src/graph/graphql.schema.json index d1177191ed..54ea1099c9 100644 --- a/projects/ui/src/graph/graphql.schema.json +++ b/projects/ui/src/graph/graphql.schema.json @@ -128883,6 +128883,18 @@ "isDeprecated": false, "deprecationReason": null }, + { + "name": "proposalsCount30d", + "description": null, + "args": [], + "type": { + "kind": "SCALAR", + "name": "Int", + "ofType": null + }, + "isDeprecated": false, + "deprecationReason": null + }, { "name": "rank", "description": null, diff --git a/projects/ui/src/graph/schema-snapshot1.graphql b/projects/ui/src/graph/schema-snapshot1.graphql index 232815531a..167bab37fc 100644 --- a/projects/ui/src/graph/schema-snapshot1.graphql +++ b/projects/ui/src/graph/schema-snapshot1.graphql @@ -320,6 +320,7 @@ type Space { proposalsCount: Int proposalsCount1d: Int proposalsCount7d: Int + proposalsCount30d: Int rank: Float skin: String strategies: [Strategy] diff --git a/projects/ui/src/hooks/app/useDebounce.ts b/projects/ui/src/hooks/app/useDebounce.ts new file mode 100644 index 0000000000..f6d18ef0cb --- /dev/null +++ b/projects/ui/src/hooks/app/useDebounce.ts @@ -0,0 +1,107 @@ +import { useState, useEffect, useRef, useMemo } from 'react'; +import { debounce } from 'lodash'; +import { exists } from '~/util'; + +type AddressIsh = { address: string }; +type ToStringIsh = { toString: () => string }; +type ToHumanIsh = { toHuman: () => string }; + +interface DebounceOptions { + leading?: boolean; + trailing?: boolean; + maxWait?: number; + equalityFn?: (left: T, right: T) => boolean; +} + +function defaultEqualityFn(left: T, right: T): boolean { + return left === right; +} +function addressIshEqualityFn(left: unknown, right: unknown): boolean { + return ( + (left as AddressIsh).address.toLowerCase() === + (right as AddressIsh).address.toLowerCase() + ); +} +function toStringIshEqualityFn(left: unknown, right: unknown): boolean { + return (left as ToStringIsh).toString() === (right as ToStringIsh).toString(); +} +function toHumanIshEqualityFn(left: unknown, right: unknown): boolean { + return (left as ToHumanIsh).toHuman() === (right as ToHumanIsh).toHuman(); +} + +const getDefaultEqualityFn = ( + value: T +): ((left: T, right: T) => boolean) => { + if (exists(value)) { + if (typeof value === 'object') { + if ('address' in value) { + return addressIshEqualityFn; + } + if ('toString' in value && typeof value.toString === 'function') { + return toStringIshEqualityFn; + } + if ('toHuman' in value && typeof value.toHuman === 'function') { + return toHumanIshEqualityFn; + } + } + } + return defaultEqualityFn; +}; + +/** + * Debounces a value, updating the debounced value when the input value changes. + * @param value - The value to debounce. + * @param delay - The delay in milliseconds. + * @param options - Options for the debounce function. + * - `leading`: Whether to invoke the debounced function on the leading edge of the timeout. + * - `trailing`: Whether to invoke the debounced function on the trailing edge of the timeout. + * - `maxWait`: The maximum time to wait before invoking the debounced function. + * - `equalityFn`: A function to determine if two values are equal. Important to memoize this. + * @returns The debounced value. + */ +function useDebounce( + value: T, + delay: number = 250, + options: DebounceOptions = {} +): T { + const { + leading = false, + trailing = true, + maxWait, + equalityFn = getDefaultEqualityFn(value), + } = options; + + const [debouncedValue, setDebouncedValue] = useState(value); + const previousValueRef = useRef(value); + + const debouncedFn = useMemo(() => { + const fn = debounce( + (newValue: T) => { + setDebouncedValue(newValue); + }, + delay, + { leading, trailing, maxWait } + ); + + return fn; + }, [delay, leading, trailing, maxWait]); + + useEffect(() => { + if (!equalityFn(previousValueRef.current, value)) { + previousValueRef.current = value; + debouncedFn(value); + + if (leading) { + setDebouncedValue(value); + } + } + + return () => { + debouncedFn.cancel(); + }; + }, [value, debouncedFn, equalityFn, leading]); + + return debouncedValue; +} + +export default useDebounce; diff --git a/projects/ui/src/img/tokens/bean-usdc-well-lp-logo.svg b/projects/ui/src/img/tokens/bean-usdc-well-lp-logo.svg new file mode 100644 index 0000000000..92b880af5f --- /dev/null +++ b/projects/ui/src/img/tokens/bean-usdc-well-lp-logo.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/projects/ui/src/img/tokens/bean-usdt-well-lp-logo.svg b/projects/ui/src/img/tokens/bean-usdt-well-lp-logo.svg new file mode 100644 index 0000000000..9d462dc852 --- /dev/null +++ b/projects/ui/src/img/tokens/bean-usdt-well-lp-logo.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/projects/ui/src/img/tokens/bean-wbtc-well-lp-logo.svg b/projects/ui/src/img/tokens/bean-wbtc-well-lp-logo.svg new file mode 100644 index 0000000000..af60ce358a --- /dev/null +++ b/projects/ui/src/img/tokens/bean-wbtc-well-lp-logo.svg @@ -0,0 +1,16 @@ + + + + + + + + + + + + + + + + diff --git a/projects/ui/src/img/tokens/bean-weeth-well-lp-logo.svg b/projects/ui/src/img/tokens/bean-weeth-well-lp-logo.svg new file mode 100644 index 0000000000..d92b3a6901 --- /dev/null +++ b/projects/ui/src/img/tokens/bean-weeth-well-lp-logo.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/projects/ui/src/lib/PipelineConvert/PipelineConvert.ts b/projects/ui/src/lib/PipelineConvert/PipelineConvert.ts new file mode 100644 index 0000000000..c31d8fba1f --- /dev/null +++ b/projects/ui/src/lib/PipelineConvert/PipelineConvert.ts @@ -0,0 +1,229 @@ +import { ethers } from 'ethers'; +import { + BeanstalkSDK, + BasinWell, + TokenValue, + AdvancedPipeStruct, + ERC20Token, + Clipboard, + ZeroExQuoteResponse, +} from '@beanstalk/sdk'; + +/** + * Parameters needed for PipelineConvert for the equal<->equal + * equal<>equal refering to + * - remove liquidity in equal proportions from source Well + * - add liquidity in equal proportions to target Well + */ +export interface BuildPipeCallArgsEqual { + sdk: BeanstalkSDK; + source: { + well: BasinWell; + lpAmountIn: TokenValue; + beanAmountOut: TokenValue; + nonBeanAmountOut: TokenValue; + }; + swap: { + buyToken: ERC20Token; + sellToken: ERC20Token; + // amount from 0xQuote.buyAmount + buyAmount: TokenValue; + // 0x quote.allowanceTarget + quote: ZeroExQuoteResponse; + }; + target: { + well: BasinWell; + amountOut: TokenValue; + }; + slippage: number; +} + +export class PipelineConvertUtil { + private static erc20Approve( + token: ERC20Token, + spender: string, + amount: ethers.BigNumberish = ethers.constants.MaxUint256, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: token.address, + callData: token + .getContract() + .interface.encodeFunctionData('approve', [spender, amount]), + clipboard, + }; + } + + private static getRemoveLiquidityEqual( + sourceWell: BasinWell, + amountIn: TokenValue, + minAmountsOut: TokenValue[], + recipient: string, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: sourceWell.address, + callData: sourceWell + .getContract() + .interface.encodeFunctionData('removeLiquidity', [ + amountIn.toBigNumber(), + minAmountsOut.map((a) => a.toBigNumber()), + recipient, + ethers.constants.MaxUint256, + ]), + clipboard, + }; + } + + private static wellSync( + well: BasinWell, + recipient: string, + amount: ethers.BigNumberish, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: well.address, + callData: well + .getContract() + .interface.encodeFunctionData('sync', [recipient, amount]), + clipboard, + }; + } + + private static transferToken( + token: ERC20Token, + recipient: string, + amount: ethers.BigNumberish, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: token.address, + callData: token + .getContract() + .interface.encodeFunctionData('transfer', [recipient, amount]), + clipboard, + }; + } + + private static junctionGte( + junction: BeanstalkSDK['contracts']['junction'], + left: ethers.BigNumberish, + right: ethers.BigNumberish, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: junction.address, + callData: junction.interface.encodeFunctionData('gte', [left, right]), + clipboard: clipboard, + }; + } + + private static junctionCheck( + junction: BeanstalkSDK['contracts']['junction'], + value: boolean, + clipboard: string = ethers.constants.HashZero + ): AdvancedPipeStruct { + return { + target: junction.address, + callData: junction.interface.encodeFunctionData('check', [value]), + clipboard, + }; + } + + static buildEqual2Equal({ + sdk, + source, + swap, + target, + slippage, + }: BuildPipeCallArgsEqual): AdvancedPipeStruct[] { + const pipe: AdvancedPipeStruct[] = []; + + const sourceWellAmountsOut = [ + source.beanAmountOut, + source.nonBeanAmountOut, + ]; + if (!source.well.tokens[0].equals(sdk.tokens.BEAN)) { + sourceWellAmountsOut.reverse(); + } + + // 0. Approve source well to spend LP tokens + pipe.push( + PipelineConvertUtil.erc20Approve(source.well.lpToken, source.well.address) + ); + + // 1. Remove liquidity from source well & set recipient to pipeline + pipe.push( + PipelineConvertUtil.getRemoveLiquidityEqual( + source.well, + source.lpAmountIn, + sourceWellAmountsOut.map((a) => a.subSlippage(slippage)), + sdk.contracts.pipeline.address + ) + ); + + // 2. Approve 0x + pipe.push( + PipelineConvertUtil.erc20Approve(swap.sellToken, swap.quote.allowanceTarget) + ); + + // 3. Swap nonBeanToken1 for nonBeanToken2. recipient MUST be Pipeline or this will fail. + pipe.push({ + target: swap.quote.to, + callData: swap.quote.data, + clipboard: Clipboard.encode([]), + }); + + // 4. transfer BuyToken to target well + pipe.push( + PipelineConvertUtil.transferToken( + swap.buyToken, + target.well.address, + ethers.constants.Zero, + Clipboard.encodeSlot(3, 0, 1) + ) + ); + + // 5. Transfer well.tokens[0] to target well + pipe.push( + PipelineConvertUtil.transferToken( + sdk.tokens.BEAN, + target.well.address, + ethers.constants.Zero, + Clipboard.encodeSlot(1, 3, 1) + ) + ); + + const minLPOut = target.amountOut.subSlippage(slippage).toBigNumber(); + + // 6. Call Sync on target well + pipe.push( + PipelineConvertUtil.wellSync( + target.well, + sdk.contracts.pipeline.address, + minLPOut + ) + ); + + // 7. Check if amount receieved from sync >= minLPOut + pipe.push( + PipelineConvertUtil.junctionGte( + sdk.contracts.junction, + ethers.constants.Zero, + minLPOut, + Clipboard.encodeSlot(6, 0, 0) + ) + ); + + // 8. Check 7 is true + pipe.push( + PipelineConvertUtil.junctionCheck( + sdk.contracts.junction, + true, + Clipboard.encodeSlot(7, 0, 0) + ) + ); + + return pipe; + } +} diff --git a/projects/ui/src/util/tenderly/api.ts b/projects/ui/src/util/tenderly/api.ts new file mode 100644 index 0000000000..ed08cec4a9 --- /dev/null +++ b/projects/ui/src/util/tenderly/api.ts @@ -0,0 +1,128 @@ +import { ChainResolver } from '@beanstalk/sdk-core'; +import { + TenderlySimulatePayload, + TenderlySimulateTxnParams, + TenderlyVnetSimulationPayload, +} from './types'; + +const TENDERLY_API_KEY = import.meta.env.VITE_TENDERLY_ACCESS_KEY; + +const TENDERLY_ACCOUNT_SLUG = import.meta.env.VITE_TENDERLY_ACCOUNT_SLUG; + +const TENDERLY_PROJECT_SLUG = import.meta.env.VITE_TENDERLY_PROJECT_SLUG; + +const TENDERLY_VNET_ID = import.meta.env.VITE_TENDERLY_VNET_ID; + +const isDevMode = import.meta.env.DEV; + +const baseEndpoint = 'https://api.tenderly.co/api/v1/account/'; + +const tenderlyEndpoint = `${baseEndpoint}/${TENDERLY_ACCOUNT_SLUG}/project/${TENDERLY_PROJECT_SLUG}`; + +const testnetEndpoint = `${tenderlyEndpoint}/vnetId/${TENDERLY_VNET_ID}`; + +type RequestInit = { + signal?: AbortSignal; +}; + +const baseHeaders = new Headers({ + 'Content-Type': 'application/json', + 'X-Access-Key': TENDERLY_API_KEY, +}); + +const tenderlyVnetSimulateTxn = async ( + payload: TenderlySimulateTxnParams, + requestInit?: RequestInit +) => { + const endpoint = `${testnetEndpoint}/transactions/simulate`; + + const params: TenderlyVnetSimulationPayload = { + callArgs: { + from: payload.from, + to: payload.to, + gas: payload.gas?.toString(), + data: payload.callData, + value: payload.value, + }, + }; + + if (payload.blockNumber) { + params.blockNumber = payload.blockNumber.toString(); + } + + const options = { + ...requestInit, + method: 'POST', + headers: baseHeaders, + body: JSON.stringify(params), + }; + + return fetch(endpoint, options); +}; + +const tenderlyProdSimulateTxn = async ( + payload: TenderlySimulateTxnParams, + requestInit?: RequestInit +) => { + const endpoint = `${tenderlyEndpoint}/simulate`; + + const params: TenderlySimulatePayload = { + network_id: payload.chainId.toString(), + from: payload.from, + to: payload.to, + input: payload.callData, + gas: payload.gas, + simulation_type: payload.simulationType || 'full', + value: payload.value, + block_number: payload.blockNumber, + }; + + const options = { + ...requestInit, + method: 'POST', + headers: baseHeaders, + body: JSON.stringify(params), + }; + + return fetch(endpoint, options); +}; + +export const tenderlySimulateTxn = async ( + payload: TenderlySimulateTxnParams, + requestInit?: RequestInit +) => { + const requestFn = + isDevMode && ChainResolver.isTestnet(payload.chainId) + ? tenderlyVnetSimulateTxn + : tenderlyProdSimulateTxn; + + try { + return requestFn(payload, requestInit).then((response) => { + if (!response.ok) { + throw new Error(`Error simulating transaction: ${response.status}`); + } + + const remaining = response.headers.get('X-Tdly-Limit'); + const resetTimeStamp = response.headers.get('X-Tdly-Reset-Timestamp'); + + console.debug( + `[TENDERLY]: ${remaining} requests remaining. Reset at ${resetTimeStamp}` + ); + + const data = response.json(); + + console.debug(`[TENDERLY] SUCCESS. response:`, response); + + return { + requestInfo: { + remaining, + resetTimeStamp, + }, + ...data, + }; + }); + } catch (error) { + console.error('Error simulating transaction', error); + throw error; + } +}; diff --git a/projects/ui/src/util/tenderly/index.ts b/projects/ui/src/util/tenderly/index.ts new file mode 100644 index 0000000000..f3b202c9ce --- /dev/null +++ b/projects/ui/src/util/tenderly/index.ts @@ -0,0 +1,2 @@ +export * from './types'; +export * from './api'; diff --git a/projects/ui/src/util/tenderly/types.ts b/projects/ui/src/util/tenderly/types.ts new file mode 100644 index 0000000000..b9b1347744 --- /dev/null +++ b/projects/ui/src/util/tenderly/types.ts @@ -0,0 +1,145 @@ +import { SupportedChainId } from '~/constants'; + +export interface TenderlySimulateTxnParams { + /** + * Network ID + */ + chainId: SupportedChainId; + /** + * Address initiating the transaction + */ + from: string; + /** + * Recipient address of the transaction + */ + to: string; + /** + * Call data string + */ + callData: string; + /** + * Type of simulation to run. Applicable only in production. + */ + simulationType?: 'full' | 'quick' | 'abi'; + /** + * Amount of Ether (in wei) sent along with the transaction. + */ + value?: string; + /** + * Amount of gas provided for the simulation. + */ + gas?: number; + /** + * Number of the block to be used for the simulation. + */ + blockNumber?: number; +} + +/** + * Tenderly simulate transaction payload + * + * @notes NOT for virtual testnet + * + * @see https://docs.tenderly.co/reference/api#/operations/simulateTransaction + */ +export interface TenderlySimulatePayload { + /** + * ID of the network on which the simulation is being run. + */ + network_id: string; + /** + * Address initiating the transaction + */ + from: string; + /** + * Recipient address of the transaction + */ + to: string; + /** + * Calldata string + */ + input: string; + /** + * Amount of gas provided for the simulation. + */ + gas?: number; + /** + * Number of the block to be used for the simulation. + */ + block_number?: number; + /** + * Index of the transaction within the block. + */ + transaction_index?: number; + /** + * String representation of a number that represents price of the gas in wei. + */ + gas_price?: string; + /** + * Flag that enables precise gas estimation. + */ + estimate_gas?: boolean; + /** + * Amount of Ether (in wei) sent along with the transaction. + */ + value?: string; + /** + * Flag indicating whether to save the simulation in dashboard UI. + */ + save?: boolean; + /** + * Flag indicating whether to save failed simulation in dashboard UI. + */ + save_failed?: boolean; + /** + * + */ + simulation_type: 'full' | 'quick' | 'abi'; +} + +/** + * Tenderly simulate transaction payload ONLY for virtual testnet + * + * @see https://docs.tenderly.co/reference/api#/operations/simulateTx + */ +export interface TenderlyVnetSimulationPayload { + callArgs: { + /** + * Address initiating the transaction + */ + from: string; + /** + * Recipient address of the transaction + */ + to: string; + /** + * Amount of gas provided for the simulation. + */ + gas?: string; + /** + * String representation of a number that represents price of the gas in wei. + */ + gasPrice?: string; + /** + * Amount of Ether (in wei) sent along with the transaction. + */ + data: string; + /** + * Amount of Ether (in wei) sent along with the transaction. + */ + value?: string; + }; + /** + * Number of the block to be used for the simulation. + */ + blockNumber?: string; + blockOverrides?: { + number: string; + timestamp: string; + }; + stateOverrides?: { + [address: string]: { + balance: string; + }; + }; +} diff --git a/projects/ui/vite.config.mts b/projects/ui/vite.config.mts index 4bf61b8abf..57afb664ce 100644 --- a/projects/ui/vite.config.mts +++ b/projects/ui/vite.config.mts @@ -1,4 +1,4 @@ -import { defineConfig, splitVendorChunkPlugin, UserConfig } from 'vite'; +import { defineConfig, splitVendorChunkPlugin } from 'vite'; import path from 'path'; import { createHtmlPlugin } from 'vite-plugin-html'; import react from '@vitejs/plugin-react'; @@ -41,6 +41,8 @@ const CSP = buildCSP({ '*.google-analytics.com', '*.doubleclick.net', 'https://gateway-arbitrum.network.thegraph.com', // Decentralized subgraph + '*.0x.org', // 0x API + '*.tenderly.co', // Tenderly API ], 'style-src': [ "'self'", @@ -63,7 +65,10 @@ const CSP = buildCSP({ 'https://cf-ipfs.com/', // Gov proposal images, 'https://*.ipfs.cf-ipfs.com/', ], - 'frame-src': ['https://verify.walletconnect.com/', 'https://verify.walletconnect.org'], // for walletconnect + 'frame-src': [ + 'https://verify.walletconnect.com/', + 'https://verify.walletconnect.org', + ], // for walletconnect }); // @ts-ignore @@ -129,4 +134,4 @@ export default defineConfig(({ command }) => ({ ], }, }, -})); +})); diff --git a/protocol/abi/Beanstalk.json b/protocol/abi/Beanstalk.json index c5945020b6..ffd90c0dd6 100644 --- a/protocol/abi/Beanstalk.json +++ b/protocol/abi/Beanstalk.json @@ -4450,43 +4450,6 @@ "stateMutability": "payable", "type": "function" }, - { - "anonymous": false, - "inputs": [ - { - "indexed": true, - "internalType": "address", - "name": "account", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "token", - "type": "address" - }, - { - "indexed": false, - "internalType": "int96", - "name": "stem", - "type": "int96" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "amount", - "type": "uint256" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "bdv", - "type": "uint256" - } - ], - "name": "AddDeposit", - "type": "event" - }, { "anonymous": false, "inputs": [ @@ -4704,43 +4667,6 @@ "name": "TransferBatch", "type": "event" }, - { - "anonymous": false, - "inputs": [ - { - "indexed": true, - "internalType": "address", - "name": "operator", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "sender", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "recipient", - "type": "address" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "depositId", - "type": "uint256" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "amount", - "type": "uint256" - } - ], - "name": "TransferSingle", - "type": "event" - }, { "inputs": [ { @@ -4934,6 +4860,35 @@ "stateMutability": "payable", "type": "function" }, + { + "inputs": [ + { + "internalType": "address", + "name": "well", + "type": "address" + }, + { + "internalType": "uint256[]", + "name": "reserves", + "type": "uint256[]" + }, + { + "internalType": "uint256", + "name": "lookback", + "type": "uint256" + } + ], + "name": "calculateDeltaBFromReserves", + "outputs": [ + { + "internalType": "int256", + "name": "", + "type": "int256" + } + ], + "stateMutability": "view", + "type": "function" + }, { "inputs": [ { @@ -8056,9 +8011,9 @@ "type": "bytes1" }, { - "internalType": "int24", + "internalType": "int32", "name": "deltaStalkEarnedPerSeason", - "type": "int24" + "type": "int32" }, { "internalType": "uint128", @@ -8185,6 +8140,80 @@ "stateMutability": "view", "type": "function" }, + { + "anonymous": false, + "inputs": [ + { + "indexed": true, + "internalType": "address", + "name": "account", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "indexed": false, + "internalType": "int96", + "name": "stem", + "type": "int96" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "amount", + "type": "uint256" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "bdv", + "type": "uint256" + } + ], + "name": "AddDeposit", + "type": "event" + }, + { + "anonymous": false, + "inputs": [ + { + "indexed": true, + "internalType": "address", + "name": "operator", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "from", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "to", + "type": "address" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "id", + "type": "uint256" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "value", + "type": "uint256" + } + ], + "name": "TransferSingle", + "type": "event" + }, { "inputs": [ { @@ -9102,30 +9131,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - }, - { - "internalType": "uint256", - "name": "percentOfDepositedBdv", - "type": "uint256" - } - ], - "name": "calcGaugePointsWithParams", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [ { @@ -9184,47 +9189,74 @@ "type": "function" }, { - "inputs": [], - "name": "getAverageGrownStalkPerBdv", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], + "name": "getCaseData", + "outputs": [ + { + "internalType": "bytes32", + "name": "casesData", + "type": "bytes32" + } + ], "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "getAverageGrownStalkPerBdvPerSeason", + "name": "getCases", "outputs": [ { - "internalType": "uint128", - "name": "", - "type": "uint128" + "internalType": "bytes32[144]", + "name": "cases", + "type": "bytes32[144]" } ], "stateMutability": "view", "type": "function" }, { - "inputs": [], - "name": "getBeanGaugePointsPerBdv", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], + "name": "getChangeFromCaseId", + "outputs": [ + { + "internalType": "uint32", + "name": "", + "type": "uint32" + }, + { + "internalType": "int8", + "name": "", + "type": "int8" + }, + { + "internalType": "uint80", + "name": "", + "type": "uint80" + }, + { + "internalType": "int80", + "name": "", + "type": "int80" + } + ], "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "getBeanToMaxLpGpPerBdvRatio", + "name": "getDeltaPodDemandLowerBound", "outputs": [ { "internalType": "uint256", @@ -9237,112 +9269,7 @@ }, { "inputs": [], - "name": "getBeanToMaxLpGpPerBdvRatioScaled", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getCaseData", - "outputs": [ - { - "internalType": "bytes32", - "name": "casesData", - "type": "bytes32" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getCases", - "outputs": [ - { - "internalType": "bytes32[144]", - "name": "cases", - "type": "bytes32[144]" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getChangeFromCaseId", - "outputs": [ - { - "internalType": "uint32", - "name": "", - "type": "uint32" - }, - { - "internalType": "int8", - "name": "", - "type": "int8" - }, - { - "internalType": "uint80", - "name": "", - "type": "uint80" - }, - { - "internalType": "int80", - "name": "", - "type": "int80" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getDeltaPodDemand", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getDeltaPodDemandLowerBound", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getDeltaPodDemandUpperBound", + "name": "getDeltaPodDemandUpperBound", "outputs": [ { "internalType": "uint256", @@ -9456,121 +9383,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePoints", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePointsPerBdvForToken", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "well", - "type": "address" - } - ], - "name": "getGaugePointsPerBdvForWell", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePointsWithParams", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getGrownStalkIssuedPerGp", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getGrownStalkIssuedPerSeason", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getLargestGpPerBdv", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getLargestLiqWell", @@ -9584,19 +9396,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [], - "name": "getLiquidityToSupplyRatio", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getLpToSupplyRatioLowerBound", @@ -9662,25 +9461,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - } - ], - "name": "getPodRate", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getPodRateLowerBound", @@ -9863,37 +9643,7 @@ }, { "inputs": [], - "name": "getSeedGauge", - "outputs": [ - { - "components": [ - { - "internalType": "uint128", - "name": "averageGrownStalkPerBdvPerSeason", - "type": "uint128" - }, - { - "internalType": "uint128", - "name": "beanToMaxLpGpPerBdvRatio", - "type": "uint128" - }, - { - "internalType": "bytes32[4]", - "name": "_buffer", - "type": "bytes32[4]" - } - ], - "internalType": "struct SeedGauge", - "name": "", - "type": "tuple" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getTargetSeasonsToCatchUp", + "name": "getTargetSeasonsToCatchUp", "outputs": [ { "internalType": "uint256", @@ -9904,19 +9654,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [], - "name": "getTotalBdv", - "outputs": [ - { - "internalType": "uint256", - "name": "totalBdv", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getTotalUsdLiquidity", @@ -10472,64 +10209,342 @@ "type": "int8" }, { - "indexed": false, + "indexed": false, + "internalType": "uint256", + "name": "fieldId", + "type": "uint256" + } + ], + "name": "TemperatureChange", + "type": "event" + }, + { + "inputs": [], + "name": "getShipmentRoutes", + "outputs": [ + { + "components": [ + { + "internalType": "address", + "name": "planContract", + "type": "address" + }, + { + "internalType": "bytes4", + "name": "planSelector", + "type": "bytes4" + }, + { + "internalType": "enum ShipmentRecipient", + "name": "recipient", + "type": "uint8" + }, + { + "internalType": "bytes", + "name": "data", + "type": "bytes" + } + ], + "internalType": "struct ShipmentRoute[]", + "name": "", + "type": "tuple[]" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "account", + "type": "address" + }, + { + "internalType": "enum LibTransfer.To", + "name": "mode", + "type": "uint8" + } + ], + "name": "gm", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "payable", + "type": "function" + }, + { + "inputs": [], + "name": "seasonTime", + "outputs": [ + { + "internalType": "uint32", + "name": "", + "type": "uint32" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "components": [ + { + "internalType": "address", + "name": "planContract", + "type": "address" + }, + { + "internalType": "bytes4", + "name": "planSelector", + "type": "bytes4" + }, + { + "internalType": "enum ShipmentRecipient", + "name": "recipient", + "type": "uint8" + }, + { + "internalType": "bytes", + "name": "data", + "type": "bytes" + } + ], + "internalType": "struct ShipmentRoute[]", + "name": "shipmentRoutes", + "type": "tuple[]" + } + ], + "name": "setShipmentRoutes", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [], + "name": "sunrise", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "payable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "internalType": "uint256", + "name": "percentOfDepositedBdv", + "type": "uint256" + } + ], + "name": "calcGaugePointsWithParams", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getAverageGrownStalkPerBdv", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getAverageGrownStalkPerBdvPerSeason", + "outputs": [ + { + "internalType": "uint128", + "name": "", + "type": "uint128" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanGaugePointsPerBdv", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanToMaxLpGpPerBdvRatio", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanToMaxLpGpPerBdvRatioScaled", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getDeltaPodDemand", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePoints", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePointsPerBdvForToken", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "well", + "type": "address" + } + ], + "name": "getGaugePointsPerBdvForWell", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePointsWithParams", + "outputs": [ + { "internalType": "uint256", - "name": "fieldId", + "name": "", "type": "uint256" } ], - "name": "TemperatureChange", - "type": "event" + "stateMutability": "view", + "type": "function" }, { "inputs": [], - "name": "getShipmentRoutes", + "name": "getGrownStalkIssuedPerGp", "outputs": [ { - "components": [ - { - "internalType": "address", - "name": "planContract", - "type": "address" - }, - { - "internalType": "bytes4", - "name": "planSelector", - "type": "bytes4" - }, - { - "internalType": "enum ShipmentRecipient", - "name": "recipient", - "type": "uint8" - }, - { - "internalType": "bytes", - "name": "data", - "type": "bytes" - } - ], - "internalType": "struct ShipmentRoute[]", + "internalType": "uint256", "name": "", - "type": "tuple[]" + "type": "uint256" } ], "stateMutability": "view", "type": "function" }, { - "inputs": [ - { - "internalType": "address", - "name": "account", - "type": "address" - }, + "inputs": [], + "name": "getGrownStalkIssuedPerSeason", + "outputs": [ { - "internalType": "enum LibTransfer.To", - "name": "mode", - "type": "uint8" + "internalType": "uint256", + "name": "", + "type": "uint256" } ], - "name": "gm", + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getLargestGpPerBdv", "outputs": [ { "internalType": "uint256", @@ -10537,17 +10552,17 @@ "type": "uint256" } ], - "stateMutability": "payable", + "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "seasonTime", + "name": "getLiquidityToSupplyRatio", "outputs": [ { - "internalType": "uint32", + "internalType": "uint256", "name": "", - "type": "uint32" + "type": "uint256" } ], "stateMutability": "view", @@ -10555,50 +10570,64 @@ }, { "inputs": [ + { + "internalType": "uint256", + "name": "fieldId", + "type": "uint256" + } + ], + "name": "getPodRate", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getSeedGauge", + "outputs": [ { "components": [ { - "internalType": "address", - "name": "planContract", - "type": "address" - }, - { - "internalType": "bytes4", - "name": "planSelector", - "type": "bytes4" + "internalType": "uint128", + "name": "averageGrownStalkPerBdvPerSeason", + "type": "uint128" }, { - "internalType": "enum ShipmentRecipient", - "name": "recipient", - "type": "uint8" + "internalType": "uint128", + "name": "beanToMaxLpGpPerBdvRatio", + "type": "uint128" }, { - "internalType": "bytes", - "name": "data", - "type": "bytes" + "internalType": "bytes32[4]", + "name": "_buffer", + "type": "bytes32[4]" } ], - "internalType": "struct ShipmentRoute[]", - "name": "shipmentRoutes", - "type": "tuple[]" + "internalType": "struct SeedGauge", + "name": "", + "type": "tuple" } ], - "name": "setShipmentRoutes", - "outputs": [], - "stateMutability": "nonpayable", + "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "sunrise", + "name": "getTotalBdv", "outputs": [ { "internalType": "uint256", - "name": "", + "name": "totalBdv", "type": "uint256" } ], - "stateMutability": "payable", + "stateMutability": "view", "type": "function" }, { @@ -11376,68 +11405,6 @@ "stateMutability": "nonpayable", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "owner", - "type": "address" - }, - { - "components": [ - { - "components": [ - { - "internalType": "address", - "name": "orderer", - "type": "address" - }, - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - }, - { - "internalType": "uint24", - "name": "pricePerPod", - "type": "uint24" - }, - { - "internalType": "uint256", - "name": "maxPlaceInLine", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "minFillAmount", - "type": "uint256" - } - ], - "internalType": "struct Order.PodOrder", - "name": "podOrder", - "type": "tuple" - }, - { - "internalType": "uint256", - "name": "beanAmount", - "type": "uint256" - } - ], - "internalType": "struct L1RecieverFacet.L1PodOrder[]", - "name": "orders", - "type": "tuple[]" - }, - { - "internalType": "bytes32[]", - "name": "proof", - "type": "bytes32[]" - } - ], - "name": "issuePodOrders", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, { "inputs": [ { @@ -11573,74 +11540,6 @@ "stateMutability": "pure", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "owner", - "type": "address" - }, - { - "components": [ - { - "components": [ - { - "internalType": "address", - "name": "orderer", - "type": "address" - }, - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - }, - { - "internalType": "uint24", - "name": "pricePerPod", - "type": "uint24" - }, - { - "internalType": "uint256", - "name": "maxPlaceInLine", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "minFillAmount", - "type": "uint256" - } - ], - "internalType": "struct Order.PodOrder", - "name": "podOrder", - "type": "tuple" - }, - { - "internalType": "uint256", - "name": "beanAmount", - "type": "uint256" - } - ], - "internalType": "struct L1RecieverFacet.L1PodOrder[]", - "name": "orders", - "type": "tuple[]" - }, - { - "internalType": "bytes32[]", - "name": "proof", - "type": "bytes32[]" - } - ], - "name": "verifyOrderProof", - "outputs": [ - { - "internalType": "bool", - "name": "", - "type": "bool" - } - ], - "stateMutability": "pure", - "type": "function" - }, { "inputs": [ { diff --git a/protocol/abi/MockBeanstalk.json b/protocol/abi/MockBeanstalk.json index 94170ab5d8..46a62687fb 100644 --- a/protocol/abi/MockBeanstalk.json +++ b/protocol/abi/MockBeanstalk.json @@ -4450,43 +4450,6 @@ "stateMutability": "payable", "type": "function" }, - { - "anonymous": false, - "inputs": [ - { - "indexed": true, - "internalType": "address", - "name": "account", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "token", - "type": "address" - }, - { - "indexed": false, - "internalType": "int96", - "name": "stem", - "type": "int96" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "amount", - "type": "uint256" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "bdv", - "type": "uint256" - } - ], - "name": "AddDeposit", - "type": "event" - }, { "anonymous": false, "inputs": [ @@ -4704,43 +4667,6 @@ "name": "TransferBatch", "type": "event" }, - { - "anonymous": false, - "inputs": [ - { - "indexed": true, - "internalType": "address", - "name": "operator", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "sender", - "type": "address" - }, - { - "indexed": true, - "internalType": "address", - "name": "recipient", - "type": "address" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "depositId", - "type": "uint256" - }, - { - "indexed": false, - "internalType": "uint256", - "name": "amount", - "type": "uint256" - } - ], - "name": "TransferSingle", - "type": "event" - }, { "inputs": [ { @@ -4934,6 +4860,35 @@ "stateMutability": "payable", "type": "function" }, + { + "inputs": [ + { + "internalType": "address", + "name": "well", + "type": "address" + }, + { + "internalType": "uint256[]", + "name": "reserves", + "type": "uint256[]" + }, + { + "internalType": "uint256", + "name": "lookback", + "type": "uint256" + } + ], + "name": "calculateDeltaBFromReserves", + "outputs": [ + { + "internalType": "int256", + "name": "", + "type": "int256" + } + ], + "stateMutability": "view", + "type": "function" + }, { "inputs": [ { @@ -8056,9 +8011,9 @@ "type": "bytes1" }, { - "internalType": "int24", + "internalType": "int32", "name": "deltaStalkEarnedPerSeason", - "type": "int24" + "type": "int32" }, { "internalType": "uint128", @@ -8185,6 +8140,80 @@ "stateMutability": "view", "type": "function" }, + { + "anonymous": false, + "inputs": [ + { + "indexed": true, + "internalType": "address", + "name": "account", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "indexed": false, + "internalType": "int96", + "name": "stem", + "type": "int96" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "amount", + "type": "uint256" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "bdv", + "type": "uint256" + } + ], + "name": "AddDeposit", + "type": "event" + }, + { + "anonymous": false, + "inputs": [ + { + "indexed": true, + "internalType": "address", + "name": "operator", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "from", + "type": "address" + }, + { + "indexed": true, + "internalType": "address", + "name": "to", + "type": "address" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "id", + "type": "uint256" + }, + { + "indexed": false, + "internalType": "uint256", + "name": "value", + "type": "uint256" + } + ], + "name": "TransferSingle", + "type": "event" + }, { "inputs": [ { @@ -8891,30 +8920,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - }, - { - "internalType": "uint256", - "name": "percentOfDepositedBdv", - "type": "uint256" - } - ], - "name": "calcGaugePointsWithParams", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [ { @@ -8973,47 +8978,74 @@ "type": "function" }, { - "inputs": [], - "name": "getAverageGrownStalkPerBdv", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], + "name": "getCaseData", + "outputs": [ + { + "internalType": "bytes32", + "name": "casesData", + "type": "bytes32" + } + ], "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "getAverageGrownStalkPerBdvPerSeason", + "name": "getCases", "outputs": [ { - "internalType": "uint128", - "name": "", - "type": "uint128" + "internalType": "bytes32[144]", + "name": "cases", + "type": "bytes32[144]" } ], "stateMutability": "view", "type": "function" }, { - "inputs": [], - "name": "getBeanGaugePointsPerBdv", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], + "name": "getChangeFromCaseId", + "outputs": [ + { + "internalType": "uint32", + "name": "", + "type": "uint32" + }, + { + "internalType": "int8", + "name": "", + "type": "int8" + }, + { + "internalType": "uint80", + "name": "", + "type": "uint80" + }, + { + "internalType": "int80", + "name": "", + "type": "int80" + } + ], "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "getBeanToMaxLpGpPerBdvRatio", + "name": "getDeltaPodDemandLowerBound", "outputs": [ { "internalType": "uint256", @@ -9026,7 +9058,7 @@ }, { "inputs": [], - "name": "getBeanToMaxLpGpPerBdvRatioScaled", + "name": "getDeltaPodDemandUpperBound", "outputs": [ { "internalType": "uint256", @@ -9037,67 +9069,91 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getCaseData", - "outputs": [ - { - "internalType": "bytes32", - "name": "casesData", - "type": "bytes32" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], - "name": "getCases", - "outputs": [ - { - "internalType": "bytes32[144]", - "name": "cases", - "type": "bytes32[144]" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getChangeFromCaseId", + "name": "getEvaluationParameters", "outputs": [ { - "internalType": "uint32", - "name": "", - "type": "uint32" - }, - { - "internalType": "int8", - "name": "", - "type": "int8" - }, - { - "internalType": "uint80", - "name": "", - "type": "uint80" - }, - { - "internalType": "int80", + "components": [ + { + "internalType": "uint256", + "name": "maxBeanMaxLpGpPerBdvRatio", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "minBeanMaxLpGpPerBdvRatio", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "targetSeasonsToCatchUp", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "podRateLowerBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "podRateOptimal", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "podRateUpperBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "deltaPodDemandLowerBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "deltaPodDemandUpperBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "lpToSupplyRatioUpperBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "lpToSupplyRatioOptimal", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "lpToSupplyRatioLowerBound", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "excessivePriceThreshold", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "soilCoefficientHigh", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "soilCoefficientLow", + "type": "uint256" + }, + { + "internalType": "uint256", + "name": "baseReward", + "type": "uint256" + } + ], + "internalType": "struct EvaluationParameters", "name": "", - "type": "int80" + "type": "tuple" } ], "stateMutability": "view", @@ -9105,7 +9161,7 @@ }, { "inputs": [], - "name": "getDeltaPodDemand", + "name": "getExcessivePriceThreshold", "outputs": [ { "internalType": "uint256", @@ -9118,12 +9174,12 @@ }, { "inputs": [], - "name": "getDeltaPodDemandLowerBound", + "name": "getLargestLiqWell", "outputs": [ { - "internalType": "uint256", + "internalType": "address", "name": "", - "type": "uint256" + "type": "address" } ], "stateMutability": "view", @@ -9131,7 +9187,7 @@ }, { "inputs": [], - "name": "getDeltaPodDemandUpperBound", + "name": "getLpToSupplyRatioLowerBound", "outputs": [ { "internalType": "uint256", @@ -9144,26 +9200,7 @@ }, { "inputs": [], - "name": "getExcessivePriceThreshold", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePoints", + "name": "getLpToSupplyRatioOptimal", "outputs": [ { "internalType": "uint256", @@ -9175,14 +9212,8 @@ "type": "function" }, { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePointsPerBdvForToken", + "inputs": [], + "name": "getLpToSupplyRatioUpperBound", "outputs": [ { "internalType": "uint256", @@ -9194,14 +9225,8 @@ "type": "function" }, { - "inputs": [ - { - "internalType": "address", - "name": "well", - "type": "address" - } - ], - "name": "getGaugePointsPerBdvForWell", + "inputs": [], + "name": "getMaxBeanMaxLpGpPerBdvRatio", "outputs": [ { "internalType": "uint256", @@ -9213,14 +9238,8 @@ "type": "function" }, { - "inputs": [ - { - "internalType": "address", - "name": "token", - "type": "address" - } - ], - "name": "getGaugePointsWithParams", + "inputs": [], + "name": "getMinBeanMaxLpGpPerBdvRatio", "outputs": [ { "internalType": "uint256", @@ -9233,7 +9252,7 @@ }, { "inputs": [], - "name": "getGrownStalkIssuedPerGp", + "name": "getPodRateLowerBound", "outputs": [ { "internalType": "uint256", @@ -9246,7 +9265,7 @@ }, { "inputs": [], - "name": "getGrownStalkIssuedPerSeason", + "name": "getPodRateOptimal", "outputs": [ { "internalType": "uint256", @@ -9259,7 +9278,7 @@ }, { "inputs": [], - "name": "getLargestGpPerBdv", + "name": "getPodRateUpperBound", "outputs": [ { "internalType": "uint256", @@ -9271,65 +9290,38 @@ "type": "function" }, { - "inputs": [], - "name": "getLargestLiqWell", - "outputs": [ - { - "internalType": "address", - "name": "", - "type": "address" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getLiquidityToSupplyRatio", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getLpToSupplyRatioLowerBound", + "name": "getRelBeanToMaxLpRatioChangeFromCaseId", "outputs": [ { - "internalType": "uint256", - "name": "", - "type": "uint256" + "internalType": "int80", + "name": "l", + "type": "int80" } ], "stateMutability": "view", "type": "function" }, { - "inputs": [], - "name": "getLpToSupplyRatioOptimal", - "outputs": [ + "inputs": [ { "internalType": "uint256", - "name": "", + "name": "caseId", "type": "uint256" } ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getLpToSupplyRatioUpperBound", + "name": "getRelTemperatureChangeFromCaseId", "outputs": [ { - "internalType": "uint256", - "name": "", - "type": "uint256" + "internalType": "uint32", + "name": "mt", + "type": "uint32" } ], "stateMutability": "view", @@ -9337,129 +9329,7 @@ }, { "inputs": [], - "name": "getMaxBeanMaxLpGpPerBdvRatio", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getMinBeanMaxLpGpPerBdvRatio", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - } - ], - "name": "getPodRate", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getPodRateLowerBound", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getPodRateOptimal", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getPodRateUpperBound", - "outputs": [ - { - "internalType": "uint256", - "name": "", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getRelBeanToMaxLpRatioChangeFromCaseId", - "outputs": [ - { - "internalType": "int80", - "name": "l", - "type": "int80" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [ - { - "internalType": "uint256", - "name": "caseId", - "type": "uint256" - } - ], - "name": "getRelTemperatureChangeFromCaseId", - "outputs": [ - { - "internalType": "uint32", - "name": "mt", - "type": "uint32" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getSeasonStruct", + "name": "getSeasonStruct", "outputs": [ { "components": [ @@ -9560,126 +9430,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [], - "name": "getSeedGauge", - "outputs": [ - { - "components": [ - { - "internalType": "uint128", - "name": "averageGrownStalkPerBdvPerSeason", - "type": "uint128" - }, - { - "internalType": "uint128", - "name": "beanToMaxLpGpPerBdvRatio", - "type": "uint128" - }, - { - "internalType": "bytes32[4]", - "name": "_buffer", - "type": "bytes32[4]" - } - ], - "internalType": "struct SeedGauge", - "name": "", - "type": "tuple" - } - ], - "stateMutability": "view", - "type": "function" - }, - { - "inputs": [], - "name": "getSeedGaugeSetting", - "outputs": [ - { - "components": [ - { - "internalType": "uint256", - "name": "maxBeanMaxLpGpPerBdvRatio", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "minBeanMaxLpGpPerBdvRatio", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "targetSeasonsToCatchUp", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "podRateLowerBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "podRateOptimal", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "podRateUpperBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "deltaPodDemandLowerBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "deltaPodDemandUpperBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "lpToSupplyRatioUpperBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "lpToSupplyRatioOptimal", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "lpToSupplyRatioLowerBound", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "excessivePriceThreshold", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "soilCoefficientHigh", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "soilCoefficientLow", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "baseReward", - "type": "uint256" - } - ], - "internalType": "struct EvaluationParameters", - "name": "", - "type": "tuple" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getTargetSeasonsToCatchUp", @@ -9693,19 +9443,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [], - "name": "getTotalBdv", - "outputs": [ - { - "internalType": "uint256", - "name": "totalBdv", - "type": "uint256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "inputs": [], "name": "getTotalUsdLiquidity", @@ -10318,7 +10055,298 @@ "type": "uint8" } ], - "name": "gm", + "name": "gm", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "payable", + "type": "function" + }, + { + "inputs": [], + "name": "seasonTime", + "outputs": [ + { + "internalType": "uint32", + "name": "", + "type": "uint32" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "components": [ + { + "internalType": "address", + "name": "planContract", + "type": "address" + }, + { + "internalType": "bytes4", + "name": "planSelector", + "type": "bytes4" + }, + { + "internalType": "enum ShipmentRecipient", + "name": "recipient", + "type": "uint8" + }, + { + "internalType": "bytes", + "name": "data", + "type": "bytes" + } + ], + "internalType": "struct ShipmentRoute[]", + "name": "shipmentRoutes", + "type": "tuple[]" + } + ], + "name": "setShipmentRoutes", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [], + "name": "sunrise", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "payable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "internalType": "uint256", + "name": "percentOfDepositedBdv", + "type": "uint256" + } + ], + "name": "calcGaugePointsWithParams", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getAverageGrownStalkPerBdv", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getAverageGrownStalkPerBdvPerSeason", + "outputs": [ + { + "internalType": "uint128", + "name": "", + "type": "uint128" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanGaugePointsPerBdv", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanToMaxLpGpPerBdvRatio", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getBeanToMaxLpGpPerBdvRatioScaled", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getDeltaPodDemand", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePoints", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePointsPerBdvForToken", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "well", + "type": "address" + } + ], + "name": "getGaugePointsPerBdvForWell", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + } + ], + "name": "getGaugePointsWithParams", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getGrownStalkIssuedPerGp", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getGrownStalkIssuedPerSeason", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getLargestGpPerBdv", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "getLiquidityToSupplyRatio", "outputs": [ { "internalType": "uint256", @@ -10326,68 +10354,69 @@ "type": "uint256" } ], - "stateMutability": "payable", + "stateMutability": "view", "type": "function" }, { - "inputs": [], - "name": "seasonTime", + "inputs": [ + { + "internalType": "uint256", + "name": "fieldId", + "type": "uint256" + } + ], + "name": "getPodRate", "outputs": [ { - "internalType": "uint32", + "internalType": "uint256", "name": "", - "type": "uint32" + "type": "uint256" } ], "stateMutability": "view", "type": "function" }, { - "inputs": [ + "inputs": [], + "name": "getSeedGauge", + "outputs": [ { "components": [ { - "internalType": "address", - "name": "planContract", - "type": "address" - }, - { - "internalType": "bytes4", - "name": "planSelector", - "type": "bytes4" + "internalType": "uint128", + "name": "averageGrownStalkPerBdvPerSeason", + "type": "uint128" }, { - "internalType": "enum ShipmentRecipient", - "name": "recipient", - "type": "uint8" + "internalType": "uint128", + "name": "beanToMaxLpGpPerBdvRatio", + "type": "uint128" }, { - "internalType": "bytes", - "name": "data", - "type": "bytes" + "internalType": "bytes32[4]", + "name": "_buffer", + "type": "bytes32[4]" } ], - "internalType": "struct ShipmentRoute[]", - "name": "shipmentRoutes", - "type": "tuple[]" + "internalType": "struct SeedGauge", + "name": "", + "type": "tuple" } ], - "name": "setShipmentRoutes", - "outputs": [], - "stateMutability": "nonpayable", + "stateMutability": "view", "type": "function" }, { "inputs": [], - "name": "sunrise", + "name": "getTotalBdv", "outputs": [ { "internalType": "uint256", - "name": "", + "name": "totalBdv", "type": "uint256" } ], - "stateMutability": "payable", + "stateMutability": "view", "type": "function" }, { @@ -11165,68 +11194,6 @@ "stateMutability": "nonpayable", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "owner", - "type": "address" - }, - { - "components": [ - { - "components": [ - { - "internalType": "address", - "name": "orderer", - "type": "address" - }, - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - }, - { - "internalType": "uint24", - "name": "pricePerPod", - "type": "uint24" - }, - { - "internalType": "uint256", - "name": "maxPlaceInLine", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "minFillAmount", - "type": "uint256" - } - ], - "internalType": "struct Order.PodOrder", - "name": "podOrder", - "type": "tuple" - }, - { - "internalType": "uint256", - "name": "beanAmount", - "type": "uint256" - } - ], - "internalType": "struct L1RecieverFacet.L1PodOrder[]", - "name": "orders", - "type": "tuple[]" - }, - { - "internalType": "bytes32[]", - "name": "proof", - "type": "bytes32[]" - } - ], - "name": "issuePodOrders", - "outputs": [], - "stateMutability": "nonpayable", - "type": "function" - }, { "inputs": [ { @@ -11362,74 +11329,6 @@ "stateMutability": "pure", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "owner", - "type": "address" - }, - { - "components": [ - { - "components": [ - { - "internalType": "address", - "name": "orderer", - "type": "address" - }, - { - "internalType": "uint256", - "name": "fieldId", - "type": "uint256" - }, - { - "internalType": "uint24", - "name": "pricePerPod", - "type": "uint24" - }, - { - "internalType": "uint256", - "name": "maxPlaceInLine", - "type": "uint256" - }, - { - "internalType": "uint256", - "name": "minFillAmount", - "type": "uint256" - } - ], - "internalType": "struct Order.PodOrder", - "name": "podOrder", - "type": "tuple" - }, - { - "internalType": "uint256", - "name": "beanAmount", - "type": "uint256" - } - ], - "internalType": "struct L1RecieverFacet.L1PodOrder[]", - "name": "orders", - "type": "tuple[]" - }, - { - "internalType": "bytes32[]", - "name": "proof", - "type": "bytes32[]" - } - ], - "name": "verifyOrderProof", - "outputs": [ - { - "internalType": "bool", - "name": "", - "type": "bool" - } - ], - "stateMutability": "pure", - "type": "function" - }, { "inputs": [ { @@ -12157,35 +12056,6 @@ "stateMutability": "view", "type": "function" }, - { - "inputs": [ - { - "internalType": "address", - "name": "well", - "type": "address" - }, - { - "internalType": "uint256[]", - "name": "reserves", - "type": "uint256[]" - }, - { - "internalType": "uint256", - "name": "lookback", - "type": "uint256" - } - ], - "name": "calculateDeltaBFromReservesE", - "outputs": [ - { - "internalType": "int256", - "name": "", - "type": "int256" - } - ], - "stateMutability": "view", - "type": "function" - }, { "anonymous": false, "inputs": [ @@ -12630,6 +12500,42 @@ "stateMutability": "nonpayable", "type": "function" }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "internalType": "uint32", + "name": "season", + "type": "uint32" + } + ], + "name": "mockSetMilestoneSeason", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "token", + "type": "address" + }, + { + "internalType": "int96", + "name": "stem", + "type": "int96" + } + ], + "name": "mockSetMilestoneStem", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, { "inputs": [], "name": "mockStepGauge",