From 60c068783ce50f58bbbbc8aa6bbb60963fe227b1 Mon Sep 17 00:00:00 2001 From: Brian Doyle Date: Fri, 22 Dec 2023 13:35:16 -0500 Subject: [PATCH] Add Building with Foundry Guide (#224) * Additional edits * Add Foundry to sidebar * Respond to feedback --- .../introduction.md | 33 +++++++------ .../building-with-base-and-foundry/testing.md | 48 +++++++++---------- apps/base-docs/sidebars.js | 24 +++++++++- 3 files changed, 64 insertions(+), 41 deletions(-) rename apps/base-docs/docs/{ => building-with-base/guides}/building-with-base-and-foundry/introduction.md (76%) rename apps/base-docs/docs/{ => building-with-base/guides}/building-with-base-and-foundry/testing.md (68%) diff --git a/apps/base-docs/docs/building-with-base-and-foundry/introduction.md b/apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/introduction.md similarity index 76% rename from apps/base-docs/docs/building-with-base-and-foundry/introduction.md rename to apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/introduction.md index f5e025347b4..fa501167ed6 100644 --- a/apps/base-docs/docs/building-with-base-and-foundry/introduction.md +++ b/apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/introduction.md @@ -5,7 +5,7 @@ keywords: [Foundry, Forge, Foundry Book, smart contract development, toolchain] hide_table_of_contents: false --- -In this article, you'll learn how to set up Foundry, the toolchain for smart contract development with Base. +In this article, you'll learn how to set up [Foundry], a toolchain for smart contract development. You'll also learn how to configure it to work with Base. --- @@ -14,18 +14,18 @@ In this article, you'll learn how to set up Foundry, the toolchain for smart con By the end of this lesson, you should be able to: - Install Foundry -- Create my first Foundry project -- Compile your first smart contract using Foundry +- Create a Foundry project +- Compile a smart contract using Foundry - Configure Foundry to work with Base ## Overview Foundry is a smart contract development toolchain that is composed of multiple small command line tools: -- *[forge]*: Compile, test, and deploy your smart contracts. -- *[cast]*: Interact with the Blockchain over RPC. You can make smart contract calls, send transactions, or retrieve any type of chain data. -- *[chisel]*: A solidity REPL. You can write solidity code directly. -- *[anvil]*: A local Blockchain node for testing and development. +- _[forge]_: Compile, test, and deploy your smart contracts +- _[cast]_: Interact with the Blockchain over RPC. You can make smart contract calls, send transactions, or retrieve any type of chain data +- _[chisel]_: A Solidity REPL. You can write Solidity code directly +- _[anvil]_: A local Blockchain node for testing and development Using Foundry you can manage your dependencies, compile your project, run tests, deploy smart contracts and interact with the chain from the command-line and via Solidity scripts. @@ -51,7 +51,7 @@ You can verify the installation by trying the following commands: ```bash $ forge --version -$ cast --version +$ cast --version $ chisel --version $ anvil --version ``` @@ -93,14 +93,18 @@ forge build: ## Setting up Foundry with Base -In order to work with `Base`, you need to configure a couple of settings in the configuration `foundry.toml` file. +In order to work with Base, you need to configure a couple of settings in the configuration `foundry.toml` file. -The first thing is the solidity version. +The first thing is the Solidity version. -> As of December 2023, Base does not support the new PUSH0 opcode introduced in Shanghai, which is the default target for the Solidity compiler if you use version 0.8.20 or later. See [Differences between Ethereum and Base] to learn more. +:::caution + +As of December 2023, Base does not support the new PUSH0 opcode introduced in Shanghai, which is the default target for the Solidity compiler if you use version 0.8.20 or later. See [Differences between Ethereum and Base] to learn more. We recommend using 0.8.19 or lower until Base is upgraded with `PUSH0`. Keep in mind that the update has been deployed to Base Goerli and Base Sepolia. +::: + You need to configure your config file as follows: ```bash @@ -137,12 +141,11 @@ We included 2 JSON RPC endpoints for `Base` and `Base goerli` and similar for th ## Conclusion -In this guide, you've embarked on the journey of smart contract development with Base and Foundry. We covered the essential steps, from installing Foundry using the convenient Foundryup toolchain installer to creating your first project and configuring Foundry to seamlessly integrate with Base. +In this guide, you've embarked on the journey of smart contract development with Base and Foundry. You've learned the essential steps, from installing Foundry using the convenient `Foundryup` toolchain installer to creating your first project and configuring Foundry to seamlessly integrate with Base. --- -## See also - +[Foundry]: https://github.com/foundry-rs/foundry [Foundry Book]: https://book.getfoundry.sh/ [chisel]: https://book.getfoundry.sh/chisel/ [cast]: https://book.getfoundry.sh/cast/ @@ -151,4 +154,4 @@ In this guide, you've embarked on the journey of smart contract development with [Git submodules]: https://git-scm.com/book/en/v2/Git-Tools-Submodules [OP Stack]: https://stack.optimism.io/ [Differences between Ethereum and Base]: https://docs.base.org/differences/ -[Basescan]: https://basescan.org/ \ No newline at end of file +[Basescan]: https://basescan.org/ diff --git a/apps/base-docs/docs/building-with-base-and-foundry/testing.md b/apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/testing.md similarity index 68% rename from apps/base-docs/docs/building-with-base-and-foundry/testing.md rename to apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/testing.md index eccb5ff3b72..14bee22db4d 100644 --- a/apps/base-docs/docs/building-with-base-and-foundry/testing.md +++ b/apps/base-docs/docs/building-with-base/guides/building-with-base-and-foundry/testing.md @@ -13,19 +13,19 @@ In this article, you'll learn how to test your smart contracts using [Foundry], By the end of this guide, you should be able to: -- Understand the importance of testing in smart contract development. -- Write and execute tests for smart contracts using the Forge Standard Library with Foundry. -- Use some of the `cheatcodes` that Foundry provides. +- Understand the increased importance of testing in smart contract development +- Write and execute tests written in Solidity using the Forge Standard Library with Foundry +- Use the `cheatcodes` that Foundry provides to test your smart contracts ## Overview -Testing is a crucial aspect of smart contract development, ensuring the reliability and security of your code. Foundry provides a robust testing framework that allows developers to create comprehensive test suites for their projects using Solidity. +Testing is a crucial aspect of smart contract development, ensuring the reliability and security of your code. Because it is impossible to patch a smart contract after deployment, you must thoroughly and completely test your code. Foundry provides a robust testing framework that allows developers to create comprehensive test suites for their projects using Solidity. ## My First Test with Foundry Consider the default test that the `forge init hello_foundry_in_base` command provides in the seed Foundry project. -```javascript +```solidity // SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.19; @@ -63,7 +63,7 @@ Take note of the following: For more information about writing tests in Foundry, you can follow the official guide for [Writing tests] -In order to run the test in Foundry, we simply run: +In order to run the test in Foundry, run: ```bash $ forge test @@ -80,22 +80,20 @@ Test result: ok. 2 passed; 0 failed; finished in 13.57ms ## Using Cheatcodes -Foundry includes a set of [Cheatcodes], which are special instructions that are accessible using the `vm` instance in your tests. Cheatcodes allows you to perform various tasks, including: +Foundry includes a set of [cheatcodes], which are special instructions that are accessible using the `vm` instance in your tests. Cheatcodes allow you to perform various tasks, including: - Manipulate the state of the blockchain - Test reverts - Test events - Change block number -- Change identity -- And more ! +- Change identity +- And more! -Let's use our first Cheatcode to modify the `msg.sender` of our tests. +To start, use a cheatcode to modify the `msg.sender` of your tests, and add some console logs via importing the `forge-std/console.sol` contract. -To do so, let's modify the smart contract to include some `console.log` that are also available in Foundry by importing the `forge-std/console.sol` contract. +The `Counter` contract should look as follows: -Our `Counter` contract should look as follows: - -```javascript +```solidity // SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.19; @@ -116,7 +114,7 @@ contract Counter { } ``` -If we run the tests using `forge test` we should see the following: +If you run the tests using `forge test`, you will see the following: ```bash Running 2 tests for test/Counter.t.sol:CounterTest @@ -125,11 +123,11 @@ Running 2 tests for test/Counter.t.sol:CounterTest Test result: ok. 2 passed; 0 failed; finished in 19.64ms ``` -It seems the logs are not being shown. The reason is because we `forge test` command includes a flag that enable us to include more details of the logs emmited during the execution of the tests. +It seems the logs are not being shown. The reason is because the `forge test` command includes a flag that enable you to include more details of the logs emitted during the execution of the tests. -We can control that by including `-vv` up until `-vvvvv`. For more details about the level of verbosity you can refer to the [Logs and Traces] section of the Foundry documentation. +You can control that by including different levels of the verbose flag -- `-vv` up to `-vvvvv`. For more details about the level of verbosity you can refer to the [Logs and Traces] section of the Foundry documentation. -If we then run the `foundry test -vv` we should see: +Run the `foundry test -vv`. You should see: ```bash Running 2 tests for test/Counter.t.sol:CounterTest @@ -145,13 +143,13 @@ Logs: Test result: ok. 2 passed; 0 failed; finished in 17.89ms ``` -Now, let's modify the test file and we will use the `prank` cheatcode, which allow us to modify the `msg.sender` of the next transaction. We will also use the `addr` cheatcode, which allow us to generate an address using any private key, which can simply be an hex number. +Now, modify the test file using `prank` cheatcode, which allow you to modify the `msg.sender` of the next transaction. You will also use the `addr` cheatcode, which allow you to generate an address using any private key, which can simply be a hex number. -We will include some `console.log` statements to understand better the execution flow. +Include some `console.log` statements to understand better the execution flow. The code should look like: -```javascript +```solidity // SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.19; @@ -183,7 +181,7 @@ contract CounterTest is Test { } ``` -Then if we run the `forge test -vv` command, we should see: +Then if you run the `forge test -vv` command, you should see: ```bash Running 2 tests for test/Counter.t.sol:CounterTest @@ -202,7 +200,7 @@ Logs: Test result: ok. 2 passed; 0 failed; finished in 48.75ms ``` -Notice how we call the cheatcode `vm.prank` before the call to the `counter.increment()` and `counter.setNumber(x)` functions. This will allow us to specify a particular address to become the `msg.sender` in the contract. Since the `vm.prank` accepts an address, we simply generate an address using the cheatcode `vm.addr`, where we pass a simple hexadecimal number, which is a valid private key. +Notice how you call the cheatcode `vm.prank` before the call to the `counter.increment()` and `counter.setNumber(x)` functions. This allows you to specify a particular address to become the `msg.sender` in the contract. Since the `vm.prank` accepts an address, you simply generate an address using the cheatcode `vm.addr`, where you pass a simple hexadecimal number, which is a valid private key. ## Conclusion @@ -216,5 +214,5 @@ Happy coding and testing with Foundry! [Foundry]: https://book.getfoundry.sh/ [Writing tests]: https://book.getfoundry.sh/forge/writing-tests -[Cheatcodes]: https://book.getfoundry.sh/forge/cheatcodes -[Logs and Traces]: https://book.getfoundry.sh/forge/tests?highlight=vvv#logs-and-traces \ No newline at end of file +[cheatcodes]: https://book.getfoundry.sh/forge/cheatcodes +[Logs and Traces]: https://book.getfoundry.sh/forge/tests?highlight=vvv#logs-and-traces diff --git a/apps/base-docs/sidebars.js b/apps/base-docs/sidebars.js index 92bb2d0b747..50360aa57d0 100644 --- a/apps/base-docs/sidebars.js +++ b/apps/base-docs/sidebars.js @@ -49,7 +49,29 @@ module.exports = { label: 'Accessing real-world data using Oracles', collapsible: true, collapsed: true, - items: ['building-with-base/guides/using-chainlink-price-feeds', 'building-with-base/guides/using-pyth-price-feeds', 'building-with-base/guides/using-supra-vrf'], + items: [ + 'building-with-base/guides/using-chainlink-price-feeds', + 'building-with-base/guides/using-pyth-price-feeds', + 'building-with-base/guides/using-supra-vrf', + ], + }, + { + type: 'category', + label: 'Building with Foundry', + collapsible: true, + collapsed: true, + items: [ + { + type: 'doc', + id: 'building-with-base/guides/building-with-base-and-foundry/introduction', + className: 'sidebar-coding', + }, + { + type: 'doc', + id: 'building-with-base/guides/building-with-base-and-foundry/testing', + className: 'sidebar-coding', + }, + ], }, { type: 'category',