First off, thanks for taking the time to contribute!
We welcome and appreciate all kinds of contributions. We ask that before contributing you please review the procedures for each type of contribution available in the Table of Contents. This will streamline the process for both maintainers and contributors. To find ways to contribute, view the I Want To Contribute section below. Larger contributions should open an issue before implementation to ensure changes don't go to waste.
We're excited to work with you and your contributions to scaling Ethereum!
- I Have a Question
- I Want To Contribute
- Reporting Bugs
- Suggesting Enhancements
- Your First Code Contribution
- Improving The Documentation
- Deploying on Devnet
- Tools
Note Before making an issue, please read the documentation and search the issues to see if your question has already been answered.
If you have any questions about the smart contracts, please feel free to ask them in the Optimism discord developer channels or create a new detailed issue.
Any and all bug reports on production smart contract code should be submitted privately to the Optimism team so that we can mitigate the issue before it is exploited. Please see our security policy document here.
- Read the documentation and the smart contracts themselves to see if the feature already exists.
- Perform a search in the issues to see if the enhancement has already been suggested. If it has, add a comment to the existing issue instead of opening a new one.
Enhancement suggestions are tracked as GitHub issues.
- Use a clear and descriptive title for the issue to identify the suggestion.
- Provide a step-by-step description of the suggested enhancement in as many details as possible.
- Describe the current behavior and why the intended behavior you expected to see differs. At this point you can also tell which alternatives do not work for you.
- Explain why this enhancement would be useful in Optimism's smart contracts. You may also want to point out the other projects that solved it better and which could serve as inspiration.
The best place to begin contributing is by looking through the issues with the good first issue
label. These are issues that are relatively easy to implement and are a great way to get familiar with the codebase.
Optimism's smart contracts are written in Solidity and we use foundry as our development framework. To get started, you'll need to install several dependencies:
- pnpm
- Make sure to
pnpm install
- foundry
- Foundry is built with rust, and this project uses a pinned version of foundry. Install the rust toolchain with
rustup
. - Make sure to install the version of foundry used by
ci-builder
, defined in the.foundryrc
file in the root of this repo. Once you havefoundryup
installed, there is a helper to do this:pnpm install:foundry
- golang
- python
Our Style Guide contains information about the project structure, syntax preferences, naming conventions, and more. Please take a look at it before submitting a PR, and let us know if you spot inconcistencies!
Once you've read the styleguide and are ready to work on your PR, there are a plethora of useful pnpm
scripts to know about that will help you with development:
pnpm build
Builds the smart contracts.pnpm test
Runs the fullforge
test suite. 1pnpm gas-snapshot
Generates the gas snapshot for the smart contracts.pnpm semver-lock
Generates the semver lockfile.pnpm storage-snapshot
Generates the storage lockfile.pnpm autogen:invariant-docs
Generates the invariant test documentation.pnpm clean
Removes all build artifacts forforge
andgo
compilations.pnpm validate-spacers
Validates the positions of the storage slot spacers.pnpm validate-deploy-configs
Validates the deployment configurations indeploy-config
pnpm slither
Runs the slither static analysis tool on the smart contracts.pnpm lint
Runs the linter on the smart contracts and scripts.pnpm pre-pr
Runs most checks, generators, and linters prior to a PR. For most PRs, this is sufficient to pass CI if everything is in order.pnpm pre-pr:full
Runs all checks, generators, and linters prior to a PR.
Documentation improvements are more than welcome! If you see a typo or feel that a code comment describes something poorly or incorrectly, please submit a PR with a fix.
To deploy the smart contracts on a local devnet, run make devnet-up
in the monorepo root. For more information on the local devnet, see devnet.md.
We use a system called "layout locking" as a safety mechanism to prevent certain contract variables from being moved to different storage slots accidentally.
To lock a contract variable, add it to the layout-lock.json
file which has the following format:
{
"MyContractName": {
"myVariableName": {
"slot": 1,
"offset": 0,
"length": 32
}
}
}
With the above config, the validate-spacers
script will check that we have a contract called MyContractName
, that the contract has a variable named myVariableName
, and that the variable is in the correct position as defined in the lock file.
You should add things to the layout-lock.json
file when you want those variables to never change.
Layout locking should be used in combination with diffing the .storage-layout
file in CI.
We use forge's gas-snapshot
subcommand to produce a gas snapshot for most tests within our suite. CI will check that the gas snapshot has been updated properly when it runs, so make sure to run pnpm gas-snapshot
!
Many of our smart contracts are semantically versioned. To make sure that changes are not made to a contract without deliberately bumping its version, we commit to the source code and the creation bytecode of its dependencies in a lockfile. Consult the Style Guide for more information about how our contracts are versioned.
Due to the many proxied contracts in Optimism's protocol, we automate tracking the diff to storage layouts of the contracts in the project. This is to ensure that we don't break a proxy by upgrading its implementation to a contract with a different storage layout. To generate the storage lockfile, run pnpm storage-snapshot
.