This document is designed to act as an outline for the process of contributing to the project. You will find some workflow tips, naming conventions, and other useful information. A standard process will increase productivity.
As you begin adding code to the repository you will notice there is a lot of code. And we have a standard naming convention for folders and files that you should follow.
- Fork the repository, and keep your
master
branch the same asmessari/subgraphs
, before creating a feature branch in your repository pull upstream changes. - Create your feature branch
branch-name
. - Once you begin to push changes create a Pull Request (PR) in
messari/subgraphs
and set it as a draft until you it is ready for review.
See "Naming Conventions" for how to name your PR
- Once your PR is ready for review go ahead and click "Ready for review". As a courtesy it is nice to let the reviewer know that it is ready through a DM.
At this point you will just need to follow the "Reviewing" process until your PR is merged.
- Generally you will need to make changes, either respond to a change/comment with a question, rebuttle, or comment. Otherwise, make the change and resolve the issue.
- When you feel changes are sufficient, let the reviewer know again that the PR is ready for another round of reviews.
- Do not make a new PR for changes, this will make it harder for the reviewer to track their progress.
- This part is straight forward, and if you are reviewing you know what to do.
- It is courtesy to let the developer know that you are done with the review.
- The reviewer is generally assigned by Vincent (@this-username-is-taken)
This is an iterative process that takes time, so don't think it will always be easy.
All our subgraphs share the same build and deployment scripts. They are used during development, when deploying a subgraph, and during our CI validation pipelines. These scripts expect all subgraphs to follow a given directory structure. So for them to work without adding extra hassle, and to keep code itself organized in a consistent and predictable manner, we require all subgraphs to follow this structure.
Our CI pipelines on PRs automatically validate that this structure is followed, to enforce it and make sure things don't break in case of error. To do so we are using folderslint. It is a simple tool that allows you to define some rules and then validates that they are met. The rules are very simple, it is a set of paths that are whitelisted. These are contained at /subgraphs/.folderslintrc
.
The directory structure for a given subgraph should be as follows (this might eventually go out of date, so always refer to the code to see the actual rules):
subgraphs/
someProtocolName/
abis/
(no limitations on child directories here)
protocols/
(might have more than 1 in the case of forks)
someProtocolName/
config/
deployments/
someProtocolName-ethereum/
configurations.json
someProtocolName-arbitrum/
configurations.json
templates/
someProtocol.template.yaml
src/
(protocol/deployment specific code)
src/
(code common to all protocols in this subgraph)
Whenever there are merge conflicts you should rebase instead of merge. The reason for this is that it keeps a clean working history.
How does this happen?
The version of master
you initially branched off is not the current version. New commits have been added, and some of them conflict with your changes. Answer: rebase.
Sometimes you need to make a change based off a feature-branch
that hasn't been merged into master
yet. Once feature-branch
is merged into master
you want to rebase onto master
to clean up the branch commit history.
There are multiple ways to do this. Most of the time you will rebase onto master
, so you should make sure your master
branch is up to date with the upstream master
(ie, Messari's repo).
Go into your feature-branch and make the following call.
git pull --rebase origin master
This will initiate a rebase. Sometimes you will have to resolve conflicts. And then git add .
and git rebase --continue
or git rebase --skip
.
Following this method, after rebasing you can do the command git push --force
to update your remote repo.
In general you want to keep PRs small when possible. This way it is easier to review and if a breaking change is merged it is easier to go back and not mess up a lot of history.
To keep PRs small follow these guidelines, and use them to make educated choices about other scenarios you might run into:
- If you are formatting code outside of the scope of your PR it should be in a separate PR
- Isolate bug fixes into individual PRs, do not combine them. If they depend on each other use your judgement if they should be together or not. You can always branch off a
feature-branch
- Use a single PR for each feature (ie, a new subgraph has its own PR)
For now after a PR is reviewed Vincent (@this-username-is-taken) does the final merge into master
.
It is nice to outline the changes / fixes you made in your PR. This way the reviewer knows what to look for and what to expect.
If the change affects a subgraph you should make a link to your testing subgraph in https://okgraph.xyz/. This website is a great hub for subgraph viewing (shoutout @0xbe1).
An example of good PR heading comments:
Sometimes a PR is so small or the name is self explanatory and a descriptive comment is not necessary. See #715. Use your judgement and ask questions if you want to learn and grow as a team!
It is nice to have a consistent naming convention for pull requests. Oftentimes there are dozens of PRs out on messari/subgraphs
so being able to know exactly what a PR is is important.
PR names also drive the commit name once a PR is merged into master
. In this way it is easier to tell what was changed in each commit.
The impact level identifiers are based on semver versioning. So you can use #name
to help prefix and categorize your PR. Read more about semver changes here.
This change is generally a bug fix, and does not break anything. ie, it is backwards compatible.
This is a change that adds a new feature, but still remains backwards compatible.
This change is usually a big update as it will not be backwards compatible with the previous code.
In addition to the version impact level you can identify a change with words to be more descriptive. In some cases a change doesn't really fit into the semver system. For example, adding this to the docs.
Some other naming identifiers:
fix
- bug fixfeat
- new featurechore
- chores (like updating README.md)docs
- adding to the docsstyle
- fix folder/file names or syntax formattingrefactor
- update logic, but the program performs the same wayperf
- add performancetest
- add a test
If you find yourself wanting to name a PR with multiple of these identifiers you should probably make a PR for each one.
"identifier
(#semver-change
); subgraph-name
; description
"
Ideally you want the
description
to be short and sweet.
Examples:
- "fix(#major); compound forks; upgrade to 2.0.1 schema"
- "feat(#minor); dashboard; add tvl to front page"
- "chore(); README; update comp version"
- "fix(#patch); abracadabra; fixing issue #420"
- "docs(); contributing; add Contributing.md"
- "refactor(#patch); aave-v2-forks; update reward logic"
- "style(); uniswap; format code"
- "perf(#patch); spookyswap; find value without contract call"
Notice: some of the names don't have a #
semver
name. This is because they don't actually affect the versioning on any of the subgraphs / dashboard. A good way to know which semver identifer to use is to notice which part of the version you are updating. And it looks like this (MAJOR.MINOR.PATCH)!
When you want to make a new branch on your forked subgraphs repo you should always start from the latest upstream/master branch. However, this process requires the github ui or a number of commands. The following walks through how to setup a custom git command.
- Go to github.com and open your forked
subgraphs
repo - Sync-up with upstream master
- Go to your local
subgraphs
repo and switch tomaster
- Pull the latest changes from remote
Start by setting the upstream branch on your local repo to messari/subgraphs
git remote add upstream https://github.com/messari/subgraphs.git
Then navigate to a folder where you can keep this bash script executable (or create a new folder). A good location is ~/bin
Once you are here create a file named git-sync
and put the following text in it:
#!/usr/bin bash
git fetch upstream # get the latest changes from the remote repository
git checkout master # switch to the local (forked) master branch
git rebase upstream/master # rebase the remote upstream/master onto forked/master
git push --force # sync remote forked/master
echo "upstream/master and forked/master are in sync"
Now use chmod
to build the executable:
chmod +x git-sync
Next, you need to add this folder's location to PATH in your environment variables (if it is not already), then run which git-sync
PATH has visibility on this new executable.
export PATH=$PWD:$PATH
which git-sync
You should see the absolute path of
git-sync
if it worked properly
Restarting your terminal and navigating to your local subgraphs
project should be enough to make this work. Just call git sync
from any branch and it should sync up your local and remote project with messari/subgraphs