- Solidity (Writing Smart Contract)
- Javascript (React & Testing)
- Web3 (Blockchain Interaction)
- Truffle (Development Framework)
- Ganache-cli (For Local Blockchain)
- Alchemy (For forking the Ethereum mainnet)
- Install NodeJS, I recommend using node version 16.5.0 to avoid any potential dependency issues
- Install Truffle, In your terminal, you can check to see if you have truffle by running
truffle --version
. To install truffle runnpm i -g truffle
. - Install Ganache-cli. To see if you have ganache-cli installed, in your command line type
ganache-cli --version
. To install, in your command line typenpm install ganache-cli --global
$ npm install
In your terminal run:
ganache-cli -f wss://eth-mainnet.alchemyapi.io/v2/<Your-App-Key> -m <Your-Mnemonic-Phrase> -u 0x0e5069514a3dd613350bab01b58fd850058e5ca4 -p 7545
Replace Your-App-Key with your Alchemy Project ID located in the settings of your project. Replace Your-Mnemonic-Phrase with your own mnemonic phrase. If you don't have a mnemonic phrase to include you can omit it:
ganache-cli -f wss://eth-mainnet.alchemyapi.io/v2/<Your-App-Key> -u 0x0e5069514a3dd613350bab01b58fd850058e5ca4 -p 7545
For the -u parameter in the command, we are unlocking an address with SHIB tokens to manipulate price of SHIB/WETH in our scripts. If you plan to use a different ERC20 token, you'll need to unlock an account holding that specific ERC20 token.
Before running any scripts, you'll want to create a .env file with the following values (see .env.example):
- ALCHEMY_API_KEY=""
- ARB_FOR="0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2" (By default we are using WETH)
- ARB_AGAINST="0x95aD61b0a150d79219dCF64E1E6Cc01f0B64C4cE" (By default we are using SHIB)
- ACCOUNT="" (Account to recieve profit/execute arbitrage contract)
- PRICE_DIFFERENCE=0.50 (Difference in price between Uniswap & Sushiswap, default is 0.50%)
- UNITS=0 (Only used for price reporting)
- GAS_LIMIT=600000 (Currently a hardcoded value, may need to adjust during testing)
- GAS_PRICE=0.0093 (Currently a hardcoded value, may need to adjust during testing)
In a seperate terminal run:
$ truffle migrate --reset
$ node ./bot.js
In another terminal run:
$ node ./scripts/manipulatePrice.js
Inside the config.json file, under the PROJECT_SETTINGS object, there are 2 keys that hold a boolean value:
- isLocal
- isDeployed
Both options depend on how you wish to test the bot. By default both values are set to true. If you set isLocal to false, and then run the bot this will allow the bot to monitor swap events on the actual mainnet, instead of locally.
isDeployed's value can be set on whether you wish for the abritrage contract to be called if a potential trade is found. By default isDeployed is set to true for local testing. Ideally this is helpful if you want to monitor swaps on mainnet and you don't have a contract deployed. This will allow you to still experiment with finding potential abitrage opportunites.
For monitoring prices and detecting potential arbitrage opportunities, you do not need to deploy the contract.
Inside the config.json file, set isDeployed to false.
See step #4 in Setting Up
$ node ./bot.js
Keep in mind you'll need to wait for an actual swap event to be triggered before it checks the price.
The bot is essentially composed of 5 functions.
- main()
- checkPrice()
- determineDirection()
- determineProfitability()
- executeTrade()
The main() function monitors swap events from both Uniswap & Sushiswap.
When a swap event occurs, it calls checkPrice(), this function will log the current price of the assets on both Uniswap & Sushiswap, and return the priceDifference
Then determineDirection() is called, this will determine where we would need to buy first, then sell. This function will return an array called routerPath in main(). The array contains Uniswap & Sushiswap's router contracts. If no array is returned, this means the priceDifference returned earlier is not higher than difference
If routerPath is not null, then we move into determineProfitability(). This is where we set our conditions on whether there is a potential arbitrage or not. This function returns either true or false.
If true is returned from determineProfitability(), then we call executeTrade() where we make our call to our arbitrage contract to perform the trade. Afterwards a report is logged, and the bot resumes to monitoring for swap events.
Both the manipulatePrice.js and bot.js has been setup to easily make some modifications easy. Before the main() function in manipulatePrice.js, there will be a comment: // -- CONFIGURE VALUES HERE -- //. Below that will be some constants you'll be able to modify such as the unlocked account, and the amount of tokens you'll want that account to spent in order to manipulate price (You'll need to adjust this if you are looking to test different pairs).
For bot.js, you'd want to take a look at the function near line 132 called determineProfitability(). Inside this function we can set our conditions and do our calculations to determine whether we may have a potential profitable trade on our hands. This function is to return true if a profitable trade is possible, and false if not.
Note if you are doing an arbitrage for a different ERC20 token than the one in the provided example (WETH), then you may also need to adjust profitability reporting in the executeTrade() function.
Keep in mind, after running the scripts, specifically manipulatePrice.js, you may need to restart your ganache cli, and re-migrate contracts to properly retest.
The bot.js script uses helper functions for fetching token pair addresses, calculating price of assets, and calculating estimated returns. These functions can be found in the helper.js file inside of the helper folder.
The helper folder also has server.js which is responsible for spinning up our local server, and initialization.js which is responsible for setting up our web3 connection, configuring Uniswap/Sushiswap contracts, etc.
The current strategy implemented is only shown as an example alongside with the manipulatePrice.js script. Essentially, after we manipulate price on Uniswap, we look at the reserves on Sushiswap and determine how much SHIB we need to buy on Uniswap to 'clear' out reserves on Sushiswap. Therefore the arbitrage direction is Uniswap -> Sushiswap.
This works because Sushiswap has lower reserves than Uniswap. However, if the arbitrage direction was swapped: Sushiswap -> Uniswap, this will sometimes error out if monitoring swaps on mainnet.
This error occurs in the determineProfitability() function inside of bot.js. Currently a try/catch is implemented, so if it errors out, the bot will just resume monitoring price. Other solutions to this may be to implement a different strategy, use different ERC20 tokens, or reversing the order.# Trader-joe-bot