git is a powerful and multifaceted tool. The ability to move chunks of commits around, to label them as branches, and to have repositories in various states of sync or not, give you amazing flexilibity. They also result in it being a tool where there is way more than one way to do something. So this document is not a definitive guide to git.
Rather, this document is a short set of answers to things that some of the Surge devs keep getting asked on Discord or in GitHub issues, put in once place, for your help.
If you think there's a better way to do things in this document, that's cool! Do them! If you think the document would be better by having them included in this document, that's cool! This document literally tells you how to change it and make a pull request. If you would like to share feedback about how your way is better and this is, in some sense, the Worst Git Guide Ever, hey ,maybe use that energy to close a bug instead! And with that said, onwards!
This document talks about how to make changes to Surge, develop new features, and submit them as pull requests. It also talks about coding standards and reviews a bit. But it pre-supposes that you have built Surge once.
The directions on how to build Surge once are in our README. Before you read that document, please follow those directions, run the resulting synth, and make sure that the About screen shows it was built on your system. Those directions are quite bulletproof at this point, but if you find an error, please do let us know.
And with that, onwards!
Register a new GitHub account, log in with your new account and go to https://github.com/surge-synthesizer/surge. In the top right corner you will see a bunch of buttons, one of which says Fork
. When you click on this, you will create a new fork of Surge onto your GitHub account.
This means that you can create a new branch
in your fork
, push
some changes into the branch and later make a pull request
(PR) out of it (towards surge-synthesizer/surge
), which then Surge maintainers can review, and, when approved, it will become a part of the actual Surge codebase.
Now that you have clicked on Fork
- a brand new fork of the Surge repository is in existence. In your case, replace the example-user
in the following addresses with your actual GitHub account name. example-user/surge
does not exist and will not exist. Use <yourusernamehere>/surge
, and throw out <
and >
while at it.
Your fork will live at this URL: http://github.com/example-user/surge
The .git file that you can clone, if you so choose, is http://github.com/example-user/surge.git.
However, since you have already cloned the surge-synthesizer/surge
onto your device, you can just add it as a remote. So type in git remote add example-user https://github.com/example-user/surge.git
- again, please be sure to replace the example-user
in both portions of that command line entry with your username. After this, you can do a git remote -vv
and, if you like type git fetch example-user
(again remembering to replace example-user
with your username.
Often something will be changed in surge-synthesizer/surge
and you will also want that in your local copy. To do this, you need to merge
your upstream into the main branch. Github has good documentation on this,
but here are the steps:
git works on the concept of "remotes", other places where you can get or push commits. When you check out, you get a remote called "origin" which is the copy on github.com of the git repo now on your hard drive. You can add lots of other remotes also.
You can do lots of operations on remotes, but the two big ones are fetch
, which brings all the diffs from the remote down to you
and push
which pushes all the diffs from you to a remote.
You need to configure these remotes one time per remote. So do this:
git remote add upstream https://github.com/surge-synthesizer/surge
git remote -v
and you should see something like:
origin git@github.com:baconpaul/surge.git (fetch)
origin git@github.com:baconpaul/surge.git (push)
upstream https://github.com/surge-synthesizer/surge (fetch)
upstream https://github.com/surge-synthesizer/surge (push)
Your actions are to fetch upstream, reset your main to match upstream, and then push that out to your GitHub.
git fetch upstream # You now have the diffs in surge-synthesizer/surge
git checkout main # you now are on your copy of main
git reset upstream/main --hard # you now have integrated the upstream diffs into your copy
git push origin main # you now have pushed your updated main back to GitHub
OK, so now you want to change your local copy somehow. There is one golden rule of doing this.
Never change your local copy in the main branch! Always make a branch for development!
If you break this rule, keeping your main in sync with the upstream main is a pain. There's a way around it, but it is a bit scary.
Think of the name of the change you are going to make. Some of the devs also include a GitHub
issue number in the name. So if you are fixing automation in GitHub issue #247 you would use a name
like automation-247
. Then create that branch as follows:
git checkout main
git checkout -b automation-247
Now you can develop to your heart's desire. When you are done developing, you can commit. At this point the branch is still just on your local hard drive, so you need to push it back to origin:
git push origin automation-247
and at this point, you can make a pull request.
Think of branches as labels on commits. You can make them all the time, commit them, push them, make a
pull request, and then delete them (git branch -d my-branch-name
) if you want. Feel free to make a new
and different branch for each pull request.
Let's say that a developer says something on Discord to the tone of "I've fixed (this issue), but it is on my example-user/example-branch
branch".
In this case, example-user
will be their actual GitHub account name, and example-branch
will be the branch that they have the fix in.
Well, to do this, you just need to add example-user
as another remote, just like upstream was. So, do this:
git remote add example-user https://www.github.com/example-user/surge
then, when you want their branch, write:
git fetch example-user
git checkout example-user/example-branch
This will give you a 'detached HEAD' message. Don't worry about that, it's beyond the scope of this exercise.
Of course, to go back to your version you can just type git checkout main
.
Sigh. OK, be careful. You are about to lose all your changes in main. Back them up somehow. Then:
git fetch upstream
git checkout main
git reset --hard upstream/main
git push origin main --force
More than enough people who end up in this situation also find this StackOverflow answer to be very useful.
Often the number of commits you do in a branch is more than the number of actual changes. For instance, a typical pattern would be develop on Mac, do a commit, then test it on Windows and Linux. On Linux, if you find an error, you'd make a one-line change and commit that. Now your change, which is still "one thing", has two commits.
Or perhaps, you develop your code over a week committing regularly and all those intermediate work commits aren't ones you want to make visible. Or perhaps you develop, then format as a last step and have changes.
In any case, as maintainers, we would sooner have a small number of git commits in our pull requests.
To accomplish this, you can squash your commits into a single commit and write a good message. To do this,
you use git rebase
which is generally a terrifying command, but in interactive mode it is not as bad.
The easiest approach is, with your branch checked out and committed, to run git rebase -i main
. This will
popup a text editor with all your commits. Have one set as "pick" (usually the first one) and set the rest to
"squash". (There are other settings of course, but the use case of flattening to one commit is best done this way).
Change the others to "squash" by just editing and writing the file. Then you get a change to restate your commit
message with the default being the union of commit messages. Rewrite and save as normal.
Then git push origin my-branch --force
will send your newly based branch up to GitHub. Do a pull request and you
should see one commit with your lovely new message.
Let's say you have developed a branch and pushed it up and made a pull request and you get code review feedback. You would like to integrate that feedback, but that's going to make another commit. Do you need a new branch and a new pull request? No, absolutely not! You can update in place and keep one clean commit. Here's how you do it.
As a starting point, we have our branch that we've submitted for a pull request with one commit. You want to integrate your concerns, so you check out the branch again. At the outset, you should see this:
paul:~/dev/music/surge$ git checkout update-git-howto
Already on 'update-git-howto'
paul:~/dev/music/surge$ git cherry -v main
+ 452b43d5e78119af80c68d0b2bf2ab47291ae3fd Add a section on code review rebase
paul:~/dev/music/surge$ git status
On branch update-git-howto
One commit on a branch. Great, now you go ahead and make your changes. Code, test, etc... and then your changes are ready to commit.
paul:~/dev/music/surge$ git add doc/
paul:~/dev/music/surge$ git commit -m "Integrate Code Review Comments from @user"
[update-git-howto d41af35] Integrate Code Review Comments from @user
1 file changed, 14 insertions(+)
paul:~/dev/music/surge$ git cherry -v main
+ 452b43d5e78119af80c68d0b2bf2ab47291ae3fd Add a section on code review rebase
+ d41af35a0e4e9658a3bca6c593b5ec33c0b4848f Integrate Code Review Comments from @user
Now at this point you could push your branch (git push origin update-git-howto
) and the PR would update
and would have two commits in it. Continuous integration would run its course and voila! But the maintainers would probably
squash those commits and rewrite your commit message, so you may want to squash down to one commit. Here's how you do it.
First rebase interactively with git rebase -i main
. You will see an editor which looks like this:
pick 452b43d Add a section on code review rebase
pick d41af35 Integrate Code Review Comments from @user
# Rebase fdf935b..d41af35 onto fdf935b (2 commands)
#
What rebase is going to do is to collapse or drop commits in the path. You want to keep the first commit and squash the second (which will mean you get one commit containing the changes of both). Change the file to say this:
pick 452b43d Add a section on code review rebase
squash d41af35 Integrate Code Review Comments from @user
# Rebase fdf935b..d41af35 onto fdf935b (2 commands)
#
And then save it. You will then get an opportunity to rewrite your commit message. The default message will be the union of the message of the two commits. Edit and save it and you should see:
paul:~/dev/music/surge$ git cherry -v main
+ c9ccb49e8c7972457a22a67ee3b799bb7a55f420 Add a section on code review rebase
Note the commit hash (in this example c9cc and so on) is a new hash which wasn't in the prior commit. That makes sense, because git has made a completely new transaction for the branch which is the union, so now you just need to update your branch. Since the commit history has changed, you need to force. So:
git push origin my-branch-name --force
and the PR page will say "@user force pushed branch" and everything can proceed.
Check out this excellent video from Dylan Beattie. It may help. But its fun!