# Building Stuff on Optimistic Ethereum
Hello and welcome! If you're looking to find out what it takes to get an app up and running on layer 2, then you've come to the right place. If at any time in this process you get stuck or have questions, please reach out on discord and drop a message in the
Optimistic Ethereum is designed from the ground up to be (almost) identical to Ethereum (but cheaper). If you've built an app on Ethereum before then you're more than equipped to deploy to OP.
You'll just have to follow four basic steps:
# Writing Contracts
Due to some technical reasons, users can currently only (realistically) deploy applications to Optimistic Ethereum if they're written in Solidity. The main reason being certain EVM opcodes are banned because they can be used to cause non-determinism during a transaction result challenge.
What's a transaction result challenge?
In short, it's the mechanism by which we make sure that L2 smart contracts are as secure as L1 smart contracts.
Anyway, the point is: certain opcodes can't be used (for security reasons), so we transform them into contract calls to a special smart contract (the "Execution Manager") that provides the functionality of those opcodes instead. If you're interested, you can find a full list of the opcodes that we transform over on the EVM/OVM comparison guide.
We had a few options for how to tackle this transformation problem, but we stuck with one of the easiest: we built a slight fork of the Solidity compiler that automatically handles these transformations when you're compiling your contracts. As a result, the easiest (and pretty much the only) way to write contracts for OP right now is to use this modified version of the Solidity compiler. Which means contracts have to be written in Solidity.
On support for different languages
In theory, we can make the same modifications to the compilers for other langauges. We haven't been able to do this yet because we're a relatively small team with a lot on our plates. If this is something you'd like to help out with, please reach out to us on discord and tell us! Seriously, I'm not just shilling the discord again, we'd love to help make that a reality and discord is the best way to get in touch with us.
We know this is kind of annoying (we're sorry Vyper devs 😞) but at least Solidity is pretty awesome.
Anyway, for now you'll have to make sure to use this fork of the compiler when you're compiling your contracts. We've built some simple tools to try to make this process as painless as possible. We'll go over the details of using these tools in the next few sections. It sounds harder than it actually is.
On the compiler
Optimism's philosophy has always been to try to "minimize our diff" — to build the features we want with the least amount of code. Less code means a smaller surface for potential issues, and L2 systems are the last place you want to have issues. We had a few options for how to approach this problem but changing a few lines of code in the Solidity compiler really was one of the simplest. We're actively working on ways to safely toss out the fork of the compiler. We know it's a little annoying and we're thankful you're sticking it out with us.
# Stuff you should know
# Constructor arguments might be "unsafe"
Whenever a contract is deployed to an Optimistic Ethereum node it undergoes a process called "safety checking". This step guarantees that all those blocked opcodes we talked about earlier were correctly transformed into calls to the "Execution Manager". Unfortunately, there's no clear way to distinguish "data" from "code" inside of contract bytecode. Under certain circumstances, constructor arguments can actually be executable code.
If your constructor arguments happen to be executable code and that executable code happens to contain invalid opcodes, you'll get an error that looks something like this:
Contract creation code contains unsafe opcodes. Did you use the right compiler or pass an unsafe constructor argument?
This sort of error is most common when using constructor arguments that contain a lot of "random" bytes (like contract addresses). We recommend using a separate initialization function if you're running into this problem.
Please consider voicing your support for EIP-2327.
This EIP would introduce a new opcode (
BEGINDATA) that would make it possible for us to tell the difference between code and constructor arguments.
You can view a longer-form discussion of the EIP on the Fellowship of Ethereum Magicians forum.
# Compiler increases bytecode size by about 25%
Because of the way we transform certain opcodes into contract calls, our fork of the Solidity compiler will produce larger bytecode than the standard compiler. This is important because Optimistic Ethereum (just like Ethereum) caps contracts at 24kb because of EIP-170. Contracts that originally compile under the limit may be larger than 24kb when compiled using our fork of the Solidity compiler. As a result, you may have to break existing contracts out into slightly smaller pieces.
Although the exact size increase depends on the contents of the contract, we've found that, on average, the compiler increases bytecode size by about 25%. This means that if your contract is smaller than ~20kb when compiled with the standard Solidity compiler it should be ok when compiled with our fork.
# Compiling Contracts
You need to compile contracts when you deploy them to Ethereum. And you also need to compile them when you deploy to Optimistic Ethereum. Ez.
Because of the custom compiler, this step of the development process is the biggest departure from the standard Ethereum development flow. Depending on your development environment, you're probably going to have to make a few tweaks in order to be able to compile contracts with our fork of the compiler.
Lucky for you, we wrote some example projects that explain how to use this custom compiler for some popular Ethereum development frameworks. We currently maintain example project repositories for:
Other development frameworks
Do you use or maintain a development framework that isn't listed above? We'd love to hear from you. We'd be more than happy to help integrate this fork of the compiler into your framework. Or we could pay someone to integrate it. Or we could even pay you to integrate it. I don't know. We'll figure it out. Point is: ping us on discord and let us know how we can help.
# Testing Contracts
You should test your contracts before you deploy them. You don't technically have to do this, but you definitely should do this. Luckily, the process of testing contracts on Optimistic Ethereum is identical to the testing process on Ethereum.
Here are the basic steps:
- Run a local Optimistic Ethereum node
- Run your tests against your local node.
By running your tests against a local version of Optimistic Ethereum, you can be confident that your contracts are going to behave correctly on testnet (and eventually mainnet).
Our local Optimistic Ethereum node setup also spins up an Ethereum node on
localhost:9545 so that you easily build and test applications that communicate between Layer 1 and Layer 2.
# Common Gotchas
# Using the wrong chain ID
If you're getting an error when sending transactions, please make sure that you're using the right chain ID.
The chain ID for the local Optimistic Ethereum node is
# Need ETH to pay for transactions on your local node
Our local Optimistic Ethereum node setup will try to charge you transaction fees (like you might expect).
You can bypass the need to pay fees locally by attaching
gasPrice: 0 to your transactions.
Alternatively, you can take a look at how we fund accounts on L2 in our integration tests for some inspiration for how to do this in your own tests. If you want to deploy a contract to testnet, you'll need to use our ETH Gateway to deposit some ETH into the L2 system (more on that in a second).
# Still seeing the same bug after a patch or new release
We frequently update our software and corresponding docker images. If you reported a bug and we merged a pull request to fix it but you're still seeing the same issue, make sure to update and rebuild your local Optimistic Ethereum setup before opening up another issue:
cd optimism git fetch git checkout develop git pull origin develop yarn clean yarn build cd ops docker-compose down -v docker-compose build --parallel
# Running out of gas but receipt says you didn't use the full gas limit
Because of some technical reasons that are hard to explain here, the maximum amount of gas you can have available to your transaction is about ~1,000,000 gas below the block gas limit. If the block gas limit is 9,000,000 and you try to send a transaction with 9,000,000 gas, you'll actually only get to use about ~8,000,000 gas. The exact amount that you'll be able to use depends on the amount of data that you send along with your transaction (more data means slightly less gas available).
# Deploying Contracts
Once you've written, compiled, and tested your contracts you'll probably want to deploy them to a live network! We currently maintain two primary networks:
optimistic-kovan, our public testnet
optimistic-ethereum, our restricted mainnet
On our restricted mainnet
Why is our mainnet "restricted"? Honestly, we're just playing it safe. Unlike their L1 counterparts, L2s can't just fork if they run into a critical bug. We're in uncharted territory right now and we want to get all of our ducks in a row before we move forward. As much as we'd love to hit the big red YOLO button, we really do need to be very careful. We appreciate your patience while we work hard to get everything in place for a smooth public mainnet launch 💗.
# Optimistic Kovan
optimistic-kovan is our public testnet and it's probably where you'll be going next.
If you'd like to deploy to
optimistic-kovan, you can use the following connection details:
# Optimistic Ethereum
optimistic-ethereum is a restricted mainnet.
Although we're currently limiting contract deployment on mainnet, you might still want to connect to the network.
optimistic-ethereum connection details are:
That's pretty much it. It's a lot like Ethereum! Except a lot more... Optimistic. If you're looking for something quick and easy to tackle next, you might want to take a look at the Official™ Optimism Tutorial. If you're looking for something a little more challenging, you should consider checking out our guide to bridging L1 and L2