Repo: https://github.com/makerdao/spells-mainnet
Responsible | Stage | Deadline |
---|---|---|
Governance | Exec Sheet is created | 15:00 UTC Week 1 Tuesday |
All | Agreement is reached on the content and roles | 15:00-15:30 UTC Week 1 Tuesday |
Crafter | Spell is cleaned up (for external contributions) | 16:00 UTC Week 1 Wednesday |
External | External code is contributed via PR (if needed) | 23:59 UTC Week 1 Friday |
Governance | Exec Sheet is finalised and fully confirmed | 23:59 UTC Week 1 Friday |
Crafter | Spell is crafted (without the Exec Hash) | 16:00 UTC Week 2 Monday |
BA Labs | Announce final rate changes (if needed) | 12:00 UTC Week 2 Tuesday |
Reviewers | Spell code is reviewed (against the Exec Sheet) | 16:00 UTC Week 2 Tuesday |
Governance | Exec Doc is merged | 16:00 UTC Week 2 Tuesday |
Crafter | Spell code review is addressed, Exec Hash is added | 12:00 UTC Week 2 Wednesday |
Reviewers | Spell code is reviewed (against the Exec Doc) | 16:00 UTC Week 2 Wednesday |
Crafter | Spell is deployed, Testnet is created | 12:00 UTC Week 2 Thursday |
Reviewers | Spell deployment is approved | 16:00 UTC Week 2 Thursday |
Crafter | Spell address is published | 16:00-16:30 UTC Week 2 Thursday |
Reviewers | Spell address is confirmed | 16:00-16:30 UTC Week 2 Thursday |
Governance | Spell address is received | 16:00-16:30 UTC Week 2 Thursday |
Reviewers | Spell PR is approved | 16:00-16:30 UTC Week 2 Thursday |
Crafter | Spell PR is merged | 16:00-16:30 UTC Week 2 Thursday |
Crafter | Spell retro is started | 16:30 UTC Week 2 Thursday |
- The deadlines are only meant for better coordination and should not be prioritised over security
- If a delay is expected, responsible party should provide new realistic time estimation
- A delay in one stage completion shifts deadlines for all subsequent stages to the same amount of hours, unless spell team agrees otherwise
- Install stable Foundry version
- Find the first Foundry release that is older than 7 days from now
- Insert the release URL here:
- Install the specified version via
foundryup --version git_tag_name
Document the installation logs containing installed versions below:
- Find the first Foundry release that is older than 7 days from now
- Create new branch
- Pull
master
branch of thespells-mainnet
repo locally - Create a new branch named
YYYY-MM-DD
using the initial target date of the spell
- Pull
- Cleanup previous spell's actions
- Check previous pull requests for the cleanup patterns
- Delete unused dependencies in the
src/dependencies
folder IF applicable - Cleanup
src/test/config.sol
- Set
deployed_spell
toaddress(0)
- Set
deployed_spell_created
to0
- Set
deployed_spell_block
to0
- IF there are outstanding spells that have not been
cast
yet, add them toprevious_spells
- Set
- Cleanup
DssSpell.sol
- Remove all definitions except
description
,officeHours
,actions
- Remove all actions inside
actions
function - Remove all interface declarations
- Keep a comment regarding
Rates
- Replace Exec Doc URL and Exec Hash with
TODO
- Remove all definitions except
- Cleanup specific tests in
DssSpell.t.sol
that are expected to be used in the future (e.g.testCollateralIntegrations
,testNewIlkRegistryValues
, ...)- Do not comment out code
- Skip by adding the
skipped
modifier - Add comments to indicate required action (e.g.
// Insert new collateral address
) - Keep all tests that are already skipped (e.g.
testOSMs
,testMedianizers
) - Remove unused interface declarations
- Ensure correctness of the cleanup
- Run Tests
make test
(ormake test match=<test_name>
to inspect debug traces)
- Run Tests
- Add comments to the spell based on the relevant Exec Sheet
- Copy every Section text from the Exec Sheet as comment to the spell code
- Surround the comment by the set of dashes (e.g.
// ----- Section text -----
) - Copy every Instruction text from the Exec Sheet (e.g.
// Instruction text
) - Add newline above every Instruction text
- Copy every
Reasoning URL
andAuthority URL
from the Exec Sheet as a comment under relevant section or instruction in the spell code (depending on the row the link is present) - For every
Reasoning URL
andAuthority URL
, add prefix derived from the url itself:// Executive Vote:
IF URL starts withhttps://vote.makerdao.com/executive/
// Poll:
IF URL starts withhttps://vote.makerdao.com/polling/
// Forum:
IF URL starts withhttps://forum.makerdao.com/t/
// MIP:
IF URL starts withhttps://mips.makerdao.com/mips/details/
- IF an action in the spell doesn't have relevant instruction (e.g.: ChainLog version bump), add the explanation below prefixed with
// Note:
- IF an instruction can not be directly taken, add a comment below prefixed with
// Note:
(e.g.:// Note: see dao_resolutions variable declared above
)
- Open draft PR
- Local tests PASS via
make test
- Commit & push the cleanup (e.g.
git commit -am "Base spell"
) - CI tests PASS
- Open draft PR on
spells-mainnet
titledMainnet spell YYYY-MM-DD
whereYYYY-MM-DD
is the expected target date of the spell - Assign PR to yourself
- Local tests PASS via
- Add content based on the provided Exec Sheet
- Ensure solc version is
0.8.16
- Office hours is
true
IF spell introduces a major change that can affect external parties (e.g.: keepers are affected in case of collateral offboarding), OTHERWISE explicitly set tofalse
- Office hours value matches the Exec Sheet, OTHERWISE notify Responsible Governance Facilitator
- Add spell actions below every spell instruction according to the Exec Sheet
- Ensure spell actions match linked sources (forum posts, polls, MIPs, etc)
- IF some actions require using interfaces
- Prefer using
DssExecLib
actions where possible (to avoid adding interfaces where not required) - Avoid multi-import layout / importing from
Interfaces.sol
(see issue #69) - Prefer single import layout (e.g.
import { VatAbstract } from "dss-interfaces/dss/VatAbstract.sol";
) - Use static interfaces IF not present in
dss-interfaces
OR present indss-interfaces
but outdated OR only a few function interfaces are needed
- Prefer using
- IF new collateral is onboarded
- Deploy
Join
contract (check which one is required)- Use JoinFab to deploy
- Ensure Etherscan Verification
- Make sure AGPLv3 is specified and used
- IF flatten, consider removing
HEVM
interface artifacts
- Deploy
Clip
contract- Use ClipFab to deploy
- Ensure Etherscan Verification
- Make sure AGPLv3 is specified and used
- Deploy
Calc
contract (check which one is required)- Use CalcFab to deploy
- Note: automatically verified on etherscan
- Check if oracle deployment is required (e.g. univ3-lp-oracle, new ilk pip, ...) with responsible ecosystem actor
- Deploy
- IF addresses are used in the spell
- Use
immutable
visibility when declaring addresses usingDssExecLib.getChangelogAddress
, OTHERWISE useconstant
for statically defined addresses - Fetch addresses as type
address
and wrap withLike
suffix interfaces inline (when making calls), EXCEPTMKR
and vesting contracts - Use the DssExecLib address helpers where possible (e.g.
DssExecLib.vat()
) - Where addresses are fetched from the ChainLog, the variable name must match the value of the ChainLog key for that address (e.g.
MCD_VAT
rather thanvat
), EXCEPT where the archive pattern differs from this pattern (e.g.MKR
)
- Use
- IF new addresses need to be added to the ChainLog
- Add new addresses to the ChainLog
- Increment ChainLog version, according to the update type
- Major -> New Vat (++.0.0)
- Minor -> Core Module (DSS) Update (e.g. Flapper) (0.++.0)
- Patch -> Collateral addition or addition/modification (0.0.++)
- New addresses are added to the
addresses_mainnet.sol
- Changes are tested via
testNewOrUpdatedChainlogValues
- Adjust system values, collateral values inside
config.sol
- Ensure every spell variable is declared as public/internal
- Ensure solc version is
- Add specific tests in
DssSpell.t.sol
to have sufficient test coverage for every spell action- Test new collaterals
- Test new ilk registry values
- Test new ChainLog values
- Test DAI/MKR streams and payments, lerps
- Test the sum of all DAI/MKR payments matches the Exec Sheet
- Run tests via
make test
(ormake test match=<test_name>
to inspect debug traces)- Ensure good coverage (every spell action is tested)
- Ensure every test function is declared as
public
- IF the test needs to run, it MUST NOT have the
skipped
modifier; OTHERWISE, it MUST have theskipped
modifier - IF a new module is initialized via the spell, the tests must include
- Sanity checks of the constructor arguments
- Sanity checks of all values added/updated by the spell function
- End-to-end "happy path" interaction with the module
- Tests PASS via
make test
- Ensure
DssExecLib
address used in current spell (DssExecLib.address
) matchesdss-exec-lib
Latest Release Tag - Push committed content to already opened PR
- Make sure CI PASS
- Mark PR as "ready for review" and add reviewers
- Notify reviewers (e.g. "the spell is ready for review")
- Wait till the Exec Doc is merged
- Exec Doc checks
- Check that every action in the spell code is present in the Exec Doc
- Check that every action in the Exec Doc is present in the spell code
- Office hours value in the Exec Doc matches the spell
- Sum of all payments in the Exec Doc matches the tests
- Exec Doc Hash
- Run
make exec-hash date=YYYY-MM-DD
and update spell code accordingly - Make sure generated hash matches with the hash provided from Governance Facilitator, OTHERWISE notify Responsible Governance Facilitator
- Ensure that executive vote file name and date is correct
- community repo commit hash corresponds to latest change
- Raw GitHub URL is correct
- Ensure the URL uses commit hash that introduced last change to the Exec Doc, NOT merge commit
- IF there is no local copy of
makerdao/community
GitHub repo), run:git clone https://github.com/makerdao/community
- OTHERWISE, ensure it is pointing to the latest commit on master:
git switch master && git pull origin master
- Get the latest commit hash for the exec doc:
git log --pretty=oneline -1 -- "<LOCAL_PATH_TO_EXEC_DOC>"
- IF there is no local copy of
- Exec hash is correct (use
cast keccak -- "$(curl '$URL' -o - 2>/dev/null)"
wherewget
doesn't work) - Ensure
description
date inDssSpell.sol
matches target date inside Exec Doc
- Run
- Make sure all review comments are either addressed or explicitly answered
- Make sure all items in the Exec Sheet are confirmed, OTHERWISE notify Responsible Governance Facilitator
- Notify the reviewers (e.g. "Exec Hash is added, reviews are addressed")
- Wait for at least two "good to deploy" comments (containing local tests) from the official reviewers
- Pre-deploy setup and checks (currently via
dapptools
)- Set local environment variables (
.sethrc
)- Deployer
- Avoid using the same deployer for mainnet and testnet (to avoid deploying contracts with the same address but different sources)
-
export ETH_PASSWORD=~/.env/password.txt
-
export ETH_KEYSTORE=~/.ethereum/keystore
-
export ETH_FROM=<address>
- EIP1559
- Run
make estimate
to estimate gas -
export ETH_GAS=X
with the output of the command above + a safety margin (e.g.export ETH_GAS=6_000_000
) - Check current gas price using
seth gas-price
and setETH_GAS_PRICE
accordingly (e.g.50 gwei
) - Consider adding margin to account for spikes (e.g. current gas price 25
gwei
, 50gwei
could be set) -
export ETH_GAS_PRICE=$(seth --to-wei X gwei)
(e.g.export ETH_GAS_PRICE=25_000_000_000
) - Check current gas priority fee and set
ETH_PRIO_FEE
accordingly -
export ETH_PRIO_FEE=$(seth --to-wei X gwei)
(e.g.export ETH_PRIO_FEE=2_000_000_000
)
- Run
-
export ETH_RPC_URL=<url>
to set mainnet RPC URL -
export ETHERSCAN_API_KEY=<key>
to set Etherscan API KEY -
source .sethrc
to make env vars available
- Deployer
- Check local env
-
seth ls
-
seth chain
-
- Set local environment variables (
- Deploy spell on mainnet
-
make deploy
- Ensure
src/test/config.sol
is edited correctly-
deployed_spell: address(<deployed_spell_address>)
-
deployed_spell_created: <timestamp>
-
deployed_spell_block: <block number>
- validate the above values via
make deploy-info tx=<tx_hash>
-
- Ensure spell is verified on etherscan
- Ensure local tests PASS against deployed spell run via the deploy script
- Push auto-generated
add deployed spell info
commit
-
- Cast spell on a newly created Tenderly Testnet
- Create testnet and cast deployed spell there using
make cast-on-tenderly spell=0x...
command - Check that returned
public explorer url
is publicly accessible (e.g. using incognito browser mode) - IF
cast-on-tenderly
command is executed several times for the same spell, delete all testnets of the same name except the last one
- Create testnet and cast deployed spell there using
- Archive Spell via
make archive-spell
for the current date (ormake archive-spell date="YYYY-MM-DD"
) using Target Date inside the Exec Doc - Commit & push changes for review
- Wait for CI to PASS
- Post a comment inside the PR containing:
- Foundry installation logs containing installed versions (from above)
- A link to the deployed spell
- A link to the created Tenderly Testnet
- Notify the reviewers (e.g. "the spell was deployed")
- Wait for at least two "good to handover" comments (containing local tests) from the official reviewers
- Communicate deployed address to governance
- Write a message with Deployed Address in
new-spells
discord channel - Tag Responsible Governance Facilitator in the message with the address
- Wait until Responsible Governance Facilitator confirms handover in
new-spells
- Write a message with Deployed Address in
- Fill the rest of the Spell Crafter-related boxes in the Exec Sheet
- Pre-Merge target branch pull attack checks
- IF within last THREE commits (or last 6 weeks) spells-mainnet repo contains a maintenance PR
- Ensure the PR actions match description and look safe
- Ensure the PR did not modify files unrelated to name / description
- IF the PR modified test or deploy scripts (including CI)
- Run old test script to ensure results are the same
- IF results different, flag with Governance Facilitators
- Obtain approval of the safety of the new script from both Spell Reviewers
- IF the PR modified
DssExecLib.address
file- Obtain approval of the safety of the new address from Spell Reviewers
- Obtain approval of the safety of the new address from Governance Facilitators
- IF within last THREE commits (or last 6 weeks) spells-mainnet repo contains a maintenance PR
- Squash & Merge
- Initiate spell retrospective (inside existing spell thread in the
#govops
discord channel)- Collect any problems noticed during the spell, propose concrete improvements to make it constructive
- Prefix your message with
Initiating retro:
for clarity - IF there is nothing to discuss, post
Initiating retro: nothing to discuss from my side
- IF
MegaPoker
-related updates are present in the spell (oracles are replaced, collaterals are onboarded or offboarded, etc)- Inform EA responsible for maintaining
MegaPoker
contract - Ensure
MegaPoker
contract is updated- Coordinate with EA responsible for maintaining
MegaPoker
and TechOps - Previous code patterns are followed
- MegaPoker changes are reviewed & approved
- CI & local tests PASS
- New
MegaPoker
contract is deployed - New deployed
MegaPoker
contract address is handed to TechOps - New deployed
Megapoker
contract address is updated in the README
- Coordinate with EA responsible for maintaining
- Inform EA responsible for maintaining
- IF new collateral is onboarded
- Ensure keeper support for new onboarded collateral with TechOps
- IF new Lerp is added
- Ensure keeper support (to call
tall
daily) via dss-cron
- Ensure keeper support (to call