Summarized using AI

Git Rebase

Brooke Kuhlmann • February 04, 2021 • online • Talk

In this talk at Ruby Galaxy v0.1, Brooke Kuhlmann discusses the advantages of utilizing the Git rebase workflow over the traditional merge workflow in version control. The primary focus is on how the rebase workflow leads to cleaner, more maintainable commit histories that facilitate easier debugging and collaboration among development teams. Key points include:

  • Linear History: The rebase approach creates a linear history of commits, making it easier to read and understand the progression of changes.
  • Debugging Advantages: A linear commit history enhances the effectiveness of debugging tools like git bisect, allowing developers to pinpoint when issues were introduced more effectively.
  • Comparison of Merge and Rebase: The talk contrasts the chaotic nature of merge commits, which can create a tangled web of histories, with the streamlined structure of rebase which provides clarity in commit history.
  • Workflow Changes: Kuhlmann describes modifications needed in the standard Git workflow: substituting git pull with a fetch and rebase process to avoid merge bubbles and maintain a clean history.
  • Setting Up Git for Rebase: Practical tips are given, such as configuring Git to always rebase by default and utilizing flags that enhance the workflow, like --prune to clean up local references.
  • Interactive Rebase: Kuhlmann delves into advanced topics like interactive rebase, explaining how to reorder commits, edit messages, and squash commits to create a more coherent commit history.
  • Resolving Merge Conflicts: The techniques for handling merge conflicts during a rebase process are discussed, emphasizing the importance of aborting if conflicts become overwhelming.
  • Introducing Git Lint: The presentation also introduces Git Lint, a Ruby gem that helps ensure well-formatted Git commit messages, further improving the clarity and professionalism of a team's commit history.
  • Final Thoughts: Kuhlmann encourages the audience to practice implementing these workflows daily, creating aliases to reduce typing, and integrating tools like Git Lint to enhance their overall Git practices.

The overall conclusion is that adopting the Git rebase workflow can significantly enhance code quality and team collaboration, resulting in a legacy of well-documented commits that facilitate future development efforts.

Git Rebase
Brooke Kuhlmann • online • Talk

Date: February 04, 2021
Published: unknown
Announced: unknown

Talk Synopsis:
Git is the dominant tool for version management. Misunderstanding and misusing Git can cost development teams time, energy, and money. Few better examples exist than Git's default merge workflow which creates repositories that are hard to read, debug, and maintain. In this talk, I'll
show how to use the Rebase Workflow instead, which puts Git to work for you to produce quality code that's easy to handle and kicks your team into high gear. We'll also get a chance to look at Git
Lint (https://www.alchemists.io/projects/git-lint) which is a linter for your Git commits written in Ruby!

Ruby Galaxy v0.1

00:00:00.640 hello ruby galaxy i'm really excited to be able to talk to you about git rebase
00:00:05.920 especially the git rebase workflow and how you can use that in order to make high quality git commits but also
00:00:12.960 um construct a well reasoned implementation a ruby implementation
00:00:18.560 in which to share with your team for code review or just basically have that history for your
00:00:24.000 past self to to empower your future self so there are several benefits to the git
00:00:31.760 rebase workflow uh the first is that it's linear the commits happen one after another
00:00:36.960 which makes them very readable so you can understand how the implementation was put together and
00:00:42.079 it kind of tells a compelling story of how you got to where you are and then on top of that it makes it very
00:00:48.320 debuggable uh because you can break out the big tools like get
00:00:53.440 bisect and other tooling to understand if something went wrong where that where that happened and how
00:01:00.000 to address it really quickly before we kind of jump into this though
00:01:05.280 and we dive a little bit further i i need to do a bit of a review here on
00:01:10.640 the history between merge and rebase so if you look at
00:01:15.840 the difference here where we have merge on the left and rebase on the right and if each one of the circles is a
00:01:22.479 commit and each color is an author we can kind of walk through this together where we have a commit
00:01:28.799 happening uh here in time on on the merge and the rebates we have another commit from a a well i should say a
00:01:35.600 series of commits from another author and then two more commits from another offer author
00:01:41.119 and what's nice about this is you can see very clearly on the right that the rebase workflow is a little easier to
00:01:47.439 understand than the merge workflow both are doing the same thing
00:01:52.720 it's just that it's a lot easier to understand what's happening in a nice rebase linear kind of flowing
00:01:59.439 structure versus a merge structure that has everything preserved in time as it occurred
00:02:06.640 so what is merge well merge is exactly joining two histories together
00:02:11.840 and they're just preserving it in time indefinitely um so if you have if you take a look at this and you have
00:02:18.000 the master branch on the left hand side and you have three different authors committing code and then you have this
00:02:24.160 feature branch on the right hand side where mariana is trying to make three commits off of aussies commit
00:02:29.680 and then eventually merge it back into master after vivian that's preserved in time date times
00:02:35.920 everything is preserved as it is which is useful um but it starts to get
00:02:42.239 out of hand if you have a whole lot of merging going on across your team not just with one author in this case but multiple authors
00:02:49.360 so let's take a look at what i mean by that if we kind of look through a couple
00:02:54.959 examples here showing emerge history and how that can kind of get out of hand so if we
00:03:00.319 if we look at a semi-linear merge history right where it's fairly linear we have like little
00:03:05.680 merge bubbles uh popping out here and there but for the most part it's it's fairly
00:03:10.720 readable not not too bad but let me show you another example where
00:03:16.720 here's here's a here's a team that i once worked on that had a heavy heavy merge workflow
00:03:23.120 and as we go through time here you can see that this is getting out of hand really quickly like at this point i
00:03:29.519 can't even tell you who's doing what in the history we could potentially plot this out
00:03:34.879 print it out put it on the floor map it out and figure out who did what when but that's a whole lot of work
00:03:41.440 that we really shouldn't have to do if we were to switch to a rebase workflow versus a merge workflow
00:03:47.760 another way to look at this is basically merge like the definition says is joining two
00:03:53.680 uh histories together right well it's a big big mess and a total
00:03:59.519 nightmare to unravel debug and figure out who did what when so i want to kind of emphasize moving
00:04:05.599 away from the merge workflow and focusing on our rebase workflow uh by the way before we continue i just
00:04:11.680 want to point out those graphs that i was showing you there with the git log and the shaws and printing out those merge bubbles
00:04:18.079 that was done by using this command so here i use git log and then i pass graph the dash dash
00:04:24.479 graph option to graph out those merge bubbles and then i pretty print it so using a format of yellow for the shaw
00:04:31.840 and then and then resetting that color and then starting a green color for the committed date and then resetting that color
00:04:38.000 is how i was printing those out these slides by the way will be available online so don't worry if you're trying to capture all this um
00:04:44.479 it'll totally be available for you later but that's how that command that's how i was able to generate those
00:04:49.759 um uh get log histories for you
00:04:54.800 so now that we've talked about merge let's talk about rebase which is basically replaying commits on top of
00:05:01.919 the the master branch or whatever branch that you created your feature branch off of
00:05:07.280 in this case we still have the same authors right we still have mariana who's doing her work but she's put her
00:05:14.240 feature brands or sorry her commits ahead of everyone else and when she finally merged that into
00:05:19.840 master they just show up as the last three commits again different
00:05:24.880 authors same amount of commits but a much easier to read uh history
00:05:30.720 another way to look at this is uh and i'm i'm borrowing these uh gifts from atlassian but another way to
00:05:37.199 look at this is if the master branch is in blue you have the commits happening there and then you have a feature branch that branches off
00:05:43.280 of that and then eventually gets replayed on top of master that's essentially what's happening here
00:05:48.560 when you're doing a rebase if it helps to look at it in a different way you can just basically pretend
00:05:55.680 that the feature branch that you're working on when you finally merge it into master you're just sliding those uh commits in front of
00:06:03.039 master when you finally commit those changes so if that helps that's another way to visualize that
00:06:10.319 so let's look at a git rebase example uh over the course of many years in this case here's a repository
00:06:17.680 in which many commits been made over the series of months series of years and as we speed this up
00:06:23.199 and we go and we fly through the years here you'll see that this history using a rebase history stays linear the entire
00:06:29.280 time you know in that other slide where i was showing you the merge conflict we couldn't even get this far
00:06:34.960 that far back in time because uh the the merge bubble would be completely off the screen so here's a
00:06:41.600 good contrast to show you the difference between the two and how powerful rebase history really is for
00:06:46.960 readability debuggability and um understandability in general for
00:06:52.160 that matter so what does a merge workflow kind of look like right like let's kind of walk
00:06:58.240 through this together talk this through and then kind of contrast that with a rebase workflow
00:07:03.840 so for starters you know when you start your work at the beginning of the day you might do a git poll which is really
00:07:09.280 two commands in one by the way if you didn't know this it's get fetch and get merged that's what getpoll is doing
00:07:14.639 then you'll start your workflow right you'll do a git switch here you'll you'll create an example branch
00:07:20.319 and then you'll do tracking tracking by the way sets up uh metadata in your uh get local config for
00:07:26.240 your for your um repository that you're working in and just ensures that the example branch
00:07:31.360 that you're creating here gets tracked so that when you push to that remote branch in the future you don't have to keep typing example
00:07:37.520 each time by creating your branch and turning on tracking git will remember that and then
00:07:43.440 you don't have to type that anymore so i highly recommend turning this on if or or using this option i should say um
00:07:49.520 when you're creating new branches then you know in a typical get work workflow you'll go ahead and make some
00:07:55.599 commits you'll make a series of commits you'll keep doing that until you're satisfied with the work and you're ready to put that up to uh up for code review
00:08:03.360 basically then you'll go ahead and maybe you'll switch back to master right you'll switch back to master
00:08:09.199 you'll pull down the latest changes because you want to make sure maybe you've been working on your feature branch for an hour and you want to pull in the changes from the rest of the team
00:08:15.280 make sure you don't have any merge conflicts then you'll switch back over to your example branch you'll pull from master
00:08:21.440 to make sure that your example branch is also up to date with changes from your team and then you'll push those changes up
00:08:27.039 and get them up for code review right now during the course of this if there's been any changes from the team you would
00:08:32.240 have merge bubbles at this point in time because every time you pulled you pulled in those changes
00:08:38.320 and that created a merge bubble so let's uh now that we've looked at merge
00:08:43.839 let's actually talk about the rebrakes workflow so if we contrast what we were just
00:08:49.120 doing in the merge workflow but highlight the differences this is what changes when you do a rebase workflow so instead of doing a
00:08:55.360 good pull what we'll do is we'll do a fetch and a poll i point out the fetch because
00:09:00.480 what we can do with the fetch is we can prune it at the same time which will delete all feature branches that you have locally
00:09:06.720 that got deleted on the remote highly highly recommend doing this and i'll show you how to configure this even
00:09:11.760 further so you don't have to type this out each time then the second thing we'll do is we'll do a poll but this time we'll teach it
00:09:17.600 to rebase that's it that is how you switch from a merge workflow to a rebase workflow is all through this one simple
00:09:24.240 uh option next we'll do some work right we'll go ahead and create our feature branch make
00:09:30.000 some commits and then we'll get ready to actually push that work up like we're doing before what we'll do in this case though is
00:09:36.480 we'll switch pull to a rebase and also the pull when we switch over to our example branch will do
00:09:42.240 a rebase there as well so we'll do a rebase origin master pull in those changes but we'll we'll
00:09:48.000 basically replay our commits on top of the um on top of whatever changes came down
00:09:54.959 from master when we did the poll and then the last thing we'll do is we'll go ahead and push these changes up
00:10:01.120 but because rebase is changes the shaw's that means that we have to do a force push
00:10:06.480 um which i highly highly recommend never never doing because it can be very dangerous right you can accidentally
00:10:11.600 override work on your feature branch or maybe if you're collaborating with somebody else you might overwrite their work
00:10:18.240 what these two options here do is force with lease will ensure that uh if
00:10:25.200 changes are happening on the remote that you aren't aware of the in some case this situation can happen when you're
00:10:30.720 pairing with somebody and you didn't communicate properly and they pushed up changes to your future branch
00:10:36.000 the same branch that you're working on and you do a force push if you turn on force with lease at least
00:10:41.279 would detect that and halt that process so you could be like oh wait maybe i should go talk to my
00:10:46.480 colleague the force if includes is a new feature and we'll talk about this a little bit
00:10:52.320 more in a few slides uh in the latest version of of git
00:10:57.519 which basically just does a double check to put it very simply uh that the shaw at the tip of the
00:11:04.320 remote branch is actually in in your history currently and if not um it'll halt that i won't spend a lot
00:11:11.920 of time on this uh but i do want to emphasize turn these on or using these when you do a push with a rebase workflow it'll save
00:11:18.959 you a lot of pain and hurt in case you're collaborating with others or forgot to communicate properly
00:11:27.040 so moving on from there let's talk about a few get configuration quick tips
00:11:33.760 the first is like i was saying when we were typing this out together there's a few things you could turn on so you don't have to type so much
00:11:39.200 one of the first things to do is edit your global git configuration and set rebase to true what this does is
00:11:45.680 this prevents you so you don't have to type git pull dash respace every time you can just do a get poll like you normally do
00:11:51.600 but now it'll rebase properly which is awesome so that's how you turn that on for every
00:11:57.120 every time you pull you'll always be rebasing by the way the command line for this is down below in case you want to turn this
00:12:03.600 on through your global git configuration the other thing you could do is you can
00:12:09.200 also update your rebases to be merges the reason i point this out
00:12:14.639 is you'll need git 22.22 or higher but this helps you if you're in a hybrid
00:12:21.040 situation where you're half your team's rebasing half your team is merging and you want to do a rebase workflow because
00:12:27.519 you understand the value of that but not all your team is there yet this would allow you to rebase but also
00:12:33.440 handle those merge bubbles when they get kind of complicated hopefully you're not in a situation like
00:12:38.800 that but it doesn't hurt to have this turned on and i highly recommend doing that you can also read up on the get
00:12:44.000 documentation on how this is used as well and and i recommend spending some time there looking at it
00:12:51.440 the other thing you could do is the fetch prune right when we were doing the fetch earlier i was showing you how to do dash dash
00:12:57.120 pruned uh to prune those feature branches that you don't need locally because they've been merged into the remote branch
00:13:02.639 or the remote repository right well if you set this to true you don't have to type get fetch dash
00:13:09.200 prune each time you can just type get fetch and it'll do the printing for you because you've configured it globally
00:13:14.720 so highly recommend turning this on as well and again this is the command line that you would run in order to
00:13:21.760 set that flag to true within your global gate configuration
00:13:27.279 lastly is a git configuration the push configuration for global configuration
00:13:32.720 so uh use force includes is what i was telling you about earlier if you don't want to type that out each
00:13:38.639 time so this requires by the way sorry this requires get 2.30
00:13:44.079 and higher to actually enable but when you enable that you don't have to type this long command line out
00:13:50.320 anymore it's still kind of long but you don't have to type out the force if includes anymore because you've turned it on
00:13:56.240 so i highly recommend turning this on too it's just a nice way for git to do that detection for you and save you some time if it determines
00:14:03.760 that things are not properly synchronized up or changes have happened on the remote
00:14:09.279 branch and you're about to force push you won't override any work
00:14:14.399 and by the way this is the command line to turn that on globally if you need it
00:14:19.680 so we've talked about uh the difference between merge and rebase at this point and we've also talked about some good
00:14:25.440 configurations that you can put applied to your global configuration to help your rebase workflow
00:14:31.519 now what i want to do is actually go through interactive rebase itself and actually talk through the various commands
00:14:38.240 so the basic way to do this is just get rebase interactive master and you're off to the races that's the
00:14:44.720 beginning of rebasing your feature branch on on master in this case assuming that you created your
00:14:50.240 feature branch off a master to begin with by the way if you don't want to type in
00:14:55.839 interactive all the time you can always do dash i and that's that's the same thing just shorter
00:15:03.839 when you do that what's going to happen is with that interactive rebase you're going to see this in your editor for the first time
00:15:09.839 you're going to see interactive rebase pop up and it's going to say hey look i have three commits here i don't know what you
00:15:16.800 want me to do with those but but here's all the i'm going to show you that i'm rebasing this shaw onto
00:15:22.160 this other shaw and by the way here's all the commands you can give me in which i can do something with
00:15:28.079 we're going to walk through all those except for those last three the label reset and merge are more
00:15:34.320 specific to kind of a rebase merge hybrid and i'm not going to spend time on that today because i want
00:15:39.519 to stay highly focused on the rebase workflow but before we get into those commands i
00:15:45.360 want to pause for a moment and just talk about the various ways in which you can interactive rebase the first is you can integrate active
00:15:52.160 rebase back by a series of numbers or a series of commits i should say sorry
00:15:57.519 so in this case with the at tilde number you can go back like three commits you could go back five commits
00:16:04.160 ten commits you name it however many commits on your feature branch you want to go back you put in that number and it will
00:16:09.519 rebase to that point the one thing that you want to do though is make sure that the number is never
00:16:16.240 higher than the number of commits on your future branch what i mean by that is if you go back uh
00:16:21.519 if you have five commits on your feature branch and you go back to six commits that means the sixth commit is actually
00:16:26.560 a commit that was on master um and you'd be changing the shot at that point which means that you can't
00:16:31.920 merge your feature work into uh the master branch once uh it's been code approve or
00:16:37.440 sorry through the code review it's been approved by you by the team so just make just be aware of that when
00:16:44.800 you're rebasing just don't go back uh past uh the commit in which you actually created the branch off of
00:16:51.040 if that helps by the way you can use the at symbol um but you can also type out head if you want to i'm showing you the
00:16:57.199 shorthand but you can also type out head if you want to uh they're just synonymous and that's another way in which to go back a
00:17:02.959 certain number of commits the other thing you can do when you're interactive rebasing is go
00:17:08.640 interactive rebase by a particular branch name this is super handy when you're in a
00:17:14.079 situation where you're rebasing you've pushed up a feature branch for code review
00:17:19.120 but now you need to continue on additional feature work which is based off the stuff that you already have in code review
00:17:24.720 so you might branch off of that feature branch so you'd have a master branch a feature branch and then another feature branch
00:17:30.880 that branches off that feature branch in that case what you want to do is you don't want to rebase all the way
00:17:36.640 back to master you want to rebase off of your current feature branch off the feature branch you just created and this
00:17:42.720 is a way to do that which is super super handy the other thing you can do is like let's
00:17:48.320 say that you've released a tag and you want to actually create a feature branch off that tag and then rebase off that tag until you can get
00:17:55.280 that hotfix or patch release out this is a nice way to do that
00:18:00.640 you can also rebase by a particular shaw so in the very first slide i was showing you where you could rebase by a
00:18:06.640 particular number and go back in time well if you don't want to do the counting in your head
00:18:12.160 you could just uh select the shaw that you want to rebase back to and the interactive rebase will go to
00:18:17.520 that sha let you rebase from that point and then complete the rebase workflow so that's a really handy technique to
00:18:23.919 have you can also rebase by upstream so this special syntax here allows you to rebase by
00:18:30.400 changes that are happen upstream i personally don't use this much myself in fact i can't think of a time that i
00:18:36.559 have had to use this but it is there for you if you ever needed to do something like that and so i just
00:18:42.400 want to point that out to you the last one here is being able to rebase by root
00:18:48.000 and this is one of my favorites it is super powerful and really awesome but it does come with a
00:18:54.640 big big caveat when you rebase by root you're basically going back to the very first commit of
00:19:00.000 the entire repository and you might be thinking hey why am i even telling you this that sounds
00:19:05.120 extremely dangerous well it is on an existing code base that you've shared with your team
00:19:10.160 or you're collaborating on with others that would be disastrous you don't want to rebase to root
00:19:15.280 but in the scenario where you're creating a repository the first time maybe you're going to open source it and share it with the world
00:19:21.280 maybe you've got an internal project that you're doing some work to get into the right state before you release it for others to
00:19:26.960 collaborate within your company this is a wonderful wonderful way in which to rebase
00:19:32.240 clean up your code clean up your commits tell a compelling story and show everyone how you built up your
00:19:38.240 implementation before you share it with them i do this all the time when i'm releasing new projects or open source
00:19:44.480 work highly highly recommend but only when the repository has not been shared
00:19:50.480 with anybody would you want to use this there's also an advanced technique when
00:19:56.240 you're rebasing to basically disable the editor from popping up every time and the way to do that is basically put
00:20:02.880 git editor true in front of your rebase command and that'll i know it sounds kind of counter productive or
00:20:09.120 counterintuitive right because you're saying true you maybe want to say something like false but if you set
00:20:15.360 it to true that'll disable the editor from popping up and this comes in really handy when you're doing a lot of fix up
00:20:21.200 commits because when you're rebasing and you have fix up commits you can just let get figure out where
00:20:26.880 those fix-ups should go and don't prompt you because there's nothing for you to do as an author
00:20:32.400 so this is a really really handy technique and great to have in your arsenal when you're rebasing heavily and
00:20:37.440 really quickly so let's talk about the get rebase
00:20:42.559 editor um let's jump into these commands and kind of walk you through each one of those
00:20:47.919 and i want to focus on all of these right here today and kind of give a brief overview and show you how to use them if you've
00:20:53.679 never seen them before i also want to point out that you can write these out every time or you can
00:20:59.280 use the shorthand and just not type out pick every time you can just use p or r or e depending on whatever command you
00:21:06.240 want to feed the rebase editor so let's start with pick and actually
00:21:11.919 explain how that works it's a it it kind of it's the default thing that you'll see when you're first rebasing
00:21:17.120 it seems kind of boring but it actually has kind of a superpower uh and what i want to point out is when
00:21:23.679 you look at let's let's say that we have three commits on our feature branch and uh we want to reorder those uh
00:21:30.159 normally with pick it's kind of a no-op in the rebase editor if you just pop up those commits in your rebase
00:21:36.720 editor and you don't do anything with them you just leave them as picks that it's kind of a no op there's really no rebase happening
00:21:42.960 but the nice thing is you could actually reorder these if you want to tell a more compelling story so in this case we have an initial
00:21:48.799 implementation we added documentation and we fixed the json schema well if we want to make a a good
00:21:55.440 compelling story here kind of read like a you know turning a page in a book uh we could reorder these uh two commits here
00:22:02.559 the last and the first um just like this and now the fix happens before we added the
00:22:07.760 implementation and added the documentation so now we're telling kind of a a different story here on how the
00:22:13.760 evolution of this implementation came to be the next thing we could do is we can
00:22:19.919 also reword commits so when we reword a commit what we do is we
00:22:25.120 change pick in this case we'll pick this middle commit uh we'll change it to reword and then
00:22:30.159 what's going to happen is this going to pop up in our editor and allow us to not only reword the subject but also the
00:22:35.840 body and this is a nice way to fix like a typo
00:22:41.200 explain ourselves better in the commit message all those things before we commit the code to our master branch or
00:22:48.640 go through code review whatever this is a nice way to clean all that up so in this case we can fix the subject
00:22:56.960 we can rewrite it this way so it's a little bit more descriptive in this case we're saying exactly that we added the table of contents to the
00:23:03.120 readme which is way more descriptive than what it was and then we can also fix our subject or
00:23:08.400 our body if we wanted to too what's really nice about this is when you have a subject and a body
00:23:14.159 the subject is the what it explains what it is that you're committing and then the why explains uh why you did
00:23:20.559 what you did right it supports the subject to explain
00:23:26.400 what your thought process was at this point in time in committing this code very very very powerful
00:23:32.960 so if we use a git log here we do a pretty format of what we just reworded here's what those commit looks
00:23:38.320 like commit messages look like and you can see the middle commit is where we reworded that
00:23:43.600 commit and it has the new subject which is exactly what we want
00:23:50.000 so if we move on here and we get into a little bit more of the advanced tools
00:23:55.520 edit is one of those and i won't be able to show all of this to you today but if you
00:24:01.440 haven't noticed so far and i i apologize for not pointing this out earlier is in the footer of these slides i have
00:24:07.440 links to screencast tutorials in which you can actually dive into each one of these commands a little bit
00:24:12.799 deeper and understand them even fuller because i just don't have enough time in this slide today to actually explain
00:24:18.960 everything but i just want you to know that you can look at those screencast tutorials for additional work and this would be
00:24:24.559 one of them that you'd want to dive into a little further so i'll explain kind of at a high level
00:24:29.600 what edit is doing in this case we can change like one of these picks to an edit
00:24:34.960 and all we would do is just reword pick to edit and we're off to the races what will happen in this case though is
00:24:41.120 during the rebase git will halt at that edit shaw that you wanted to edit
00:24:47.200 and you can do a multitude of things you can fix up your implementation you can rerun your specs if you wanted
00:24:52.880 to you could edit your subject your body of the commit message you name it when you're finally happy
00:24:59.679 with those changes and you commit those then you can do a get rebase continue and continue on with the rebase and
00:25:05.440 and um you're done at that point so edit's a nice way to go back in time
00:25:11.440 in one of the commits on your future branch and just basically correct them both in terms of
00:25:16.480 commit message and implementation really really powerful tool
00:25:21.520 the other thing you could do is you can squash a commit and what squash does is in these three
00:25:26.720 commits that we've been looking at so far we could take two of those and we could actually squash those into the previous commit so
00:25:33.120 in this case we would change those two commits to squashes and then they would just squash themselves into the previous commit
00:25:39.360 now what squash does though in that case is it takes all three of those commits the subjects and the bodies from all three and
00:25:45.600 smashes them together and it wants to make one big commit message by default
00:25:50.720 you don't want to do this because as you're reading this here this is kind of a mess to read through it's it's a lot to decipher there's a
00:25:57.520 three different ideas going on because we're squashing three commits into one so what we can do is get rebase will
00:26:03.679 halt in the editor with those three commit messages all squashed together and now we can fix them up
00:26:09.600 and the way to do that is we can go ahead and kind of look at our subjects we can look at our bodies and be like
00:26:15.360 hey you know what we can actually write this a little bit better first of all we can throw out that we fix the json schema
00:26:22.000 because we're squashing them all three commits down into one commit so the fact that we had a screw-up we
00:26:28.159 could just make that magically disappear as if we hatched it perfectly the first time uh which is a nice little uh a trick
00:26:35.919 the next thing we can do is we can get rid of this middle subject um it's not really helping us in this case
00:26:41.600 we can kind of just sort of reword that a little bit so what we can do is um collapse that
00:26:47.840 down have our two subjects now and then this last body now we're starting to get kind of to the meat of things here this is
00:26:53.679 starting to tell a more compelling story so we like the subject but these two uh met these two
00:27:00.960 paragraphs here are not really working for us so let's just tear out this first line and reword this
00:27:07.039 so that it actually reads better as a paragraph and that tells a more compelling story now we have a clear subject
00:27:13.360 and we have a clear body and a nice commit message that summarizes those three commits that we just squashed down
00:27:18.880 into one commit this is a really nice way to use squash
00:27:24.000 the other thing we do is we can do a fix up now fix up is like squash but it's a little lighter weight
00:27:29.360 it's actually one of my favorite commands i use it all the time and to show you it works the same way as
00:27:36.559 squash so in this case we'll take pick well sorry we'll we'll pick a couple of these commits right
00:27:42.159 we'll make a fix up and we're going to fix up this commit into one of the other commits right so we'll move that fix up
00:27:48.799 here to the first commit and that's going to take that middle fix up commit and just fix
00:27:54.559 it into the previous command what's going to happen though is the subject and the body of the fix-up
00:28:00.960 commit is going to be thrown out in favor of taking the first commits subject and body
00:28:07.120 what we're fixing is really the implementation not the commit message in this case that's the big difference between squash and fix up
00:28:13.360 but super super handy when half the time you are just wanting to fix up your implementation
00:28:18.960 this is a great way to do that quickly without having to re-edit your commit messages each time
00:28:24.320 so now if we run a git log on the git commits after we've done our
00:28:30.000 rebase we'll see that we only have two commits instead of three commits now because we fixed the first one into the previous
00:28:35.919 command which is really really nice
00:28:40.960 so we can also execute code when we're rebasing believe it or not
00:28:46.399 execution is a nice way to run additional tests scripts maybe some diagnostic tools in
00:28:52.240 between our commits when we're doing a rebase it's a really really powerful tool
00:28:57.360 but i'll just touch on it briefly today and let you kind of experiment and go wild with it after so the way this works is exec
00:29:06.320 works a little differently than all the other commands where we've been renaming pic to whatever command we want
00:29:11.440 with exec we actually have to slip it in after the commit that we want to do something with so that middle commit that we were just
00:29:18.080 looking at we'll put the exact script after that commit and then run that script so in this case i'm doing a kind
00:29:24.559 of a very simple contrived example where i'm just going to print a string which is called this is an
00:29:29.679 example script in this case and that's going to run in the middle of the rebase right after that second commit but
00:29:36.559 before we actually finish our rebase with the fixed json schema that last command and what it'll do is it'll just spit out
00:29:43.200 that text in the command line it's just a simple print command i know this is very
00:29:48.640 simple but imagine if you wanted to run like your r-spec test suite or maybe r-spec
00:29:56.720 on that particular uh implementation or files that were changed in the previous commit or any other kind of diagnostics that
00:30:03.279 you might want to run the sky's the limit on what you can do here and also in this case i've only
00:30:08.640 shown you running a little simple script within the the get rebase editor there's nothing
00:30:15.360 stopping you from actually having a bash script and just executing the bash script here too so sky's the limit on what you could do
00:30:21.520 with this uh with this command moving on actually is another command
00:30:28.159 called break and this is recent i forget which version of git this was added in but it's a nice way to actually kind of
00:30:34.320 put a debug statement in the middle of your rebase it works a lot like exec but it doesn't actually execute anything it just
00:30:41.520 halts your rebase in the middle so again if we take this middle commit right
00:30:46.799 and we put a break after that what will happen during the course of the rebase is uh it'll start rebasing those first
00:30:53.279 two commits and then it'll halt at the break and then just kind of hang out waiting for you to continue the rebates you'd
00:30:59.679 have to do a git rebase dash dash continue at that point to continue the rebase
00:31:04.960 but break by the way is a it's just like i've been saying it's just a great way in which to kind of do a break point in
00:31:11.279 the middle of middle of your rebase maybe you want to add more commits maybe you just want to look at your implementation at the commit that it
00:31:17.679 broke broke at um you know sky's the limit on what you want to do but it's a it's a nice technique for for
00:31:24.000 debugging rebase finally the very last command is drop
00:31:29.919 and it's really simple almost as simple as pick except the only difference is is that when you
00:31:35.039 change something to a drop it's actually going to delete that commit entirely the commit message the implementation
00:31:41.679 everything with that commit is going to be deleted technically it's not uh
00:31:46.720 completely gone if you screwed up it would actually be in your ref log but i don't have enough time today to actually dive into the ref log for you
00:31:53.600 but that is what's happening is it's deleting that commit if you don't like to type a lot like i
00:32:00.559 do um you don't actually even have to type out drop in this case the other way to drop a commit is
00:32:06.720 basically just delete the line that you don't want and when you save and let rebase finish
00:32:12.080 it'll delete that command just as if you physically typed up drop which is another cool way to drop
00:32:19.279 commits that you don't want so if we look at the git log again and
00:32:24.559 after our rebase you'll see that we only have two commits left right we dropped the middle commit and that's exactly what we wanted in this case
00:32:33.120 okay so we've gone through all the commands now i want to talk about get rebased resolution this is
00:32:38.960 basically what you have to go through when you are doing a rebase and you have a merge conflict and what
00:32:45.120 do you do in that situation so the very first command is being able
00:32:50.320 to abort any point in time when you're rebasing if you get scared uh you're nervous um you're not familiar
00:32:57.039 with get rebase or maybe things are just the merge conflict is so gnarly that you don't know what to do with it
00:33:02.880 you can just abort at any time and it just uh prevents it just rolls back the commit or sorry the rebates it's as if you
00:33:09.360 didn't do the rebase to begin with so this is your escape patch a really cool command if you
00:33:14.880 in the middle of a rebase and you just want to start all over if you're also in the middle of rebase
00:33:19.919 and you hit a merge conflict and you resolve it when you want to continue the rebase this is how you continue it
00:33:26.399 same thing like i was saying before if you were doing a break or you were doing a rebase with an execution
00:33:31.600 or an edit and you needed it somehow halted the get rebase this is how you would
00:33:36.880 continue in certain circumstances when you're rebasing sometimes rebase will just stop
00:33:42.960 but it actually doesn't have anything to do it just kind of halted at a place where it resolved something but there's
00:33:48.080 nothing for you to do with the get rebase so in that case you can just happily skip and continue on with the rebase
00:33:55.760 basically just telling rebase to continue on its merry way the other thing you could do too is
00:34:01.840 maybe you're in the middle of a pretty complicated rebase like maybe there's like 10 commits on your feature branch
00:34:07.039 you had a couple merge conflicts and at a certain point you just kind of forgot where you were in the rebase uh flow well in the middle of rebase you
00:34:14.399 can always do git rebase show current patch and that'll give you the diagnostics of where you're at it'll show you what shaw
00:34:19.599 you're on it'll show you a diff of the of the current
00:34:24.720 shaw that you're on and you can kind of get a sense of where you're at which is
00:34:30.000 really nice you can always use git status as well but i kind of like git show current patch because it's a
00:34:35.359 nice way to give you a little bit more insight into the actual rebase that you're doing both will work i just kind of like this
00:34:42.079 one a lot better now we've gone through all the commands for rebasing
00:34:48.720 but there are a couple special ways that you can do rebasing a little bit differently using commands
00:34:54.000 that you're probably a little bit more familiar with which is git commit if you didn't know you can also pass
00:35:00.079 a dash fix up into git commit with the shaw of the commit you want to fix
00:35:05.280 so let me show you what that looks like so if we look at our git log and we have these three commits that
00:35:11.119 we've been talking about let's say that we want to fix up one of those like let's say that the initial implementation isn't quite
00:35:17.200 the way that we wanted it and we wanted to make a few changes to the files in that initial implementation
00:35:22.720 well we could edit those files make those changes commit them but this time
00:35:27.839 commit them not making a new commit message but instead making a fix up commit message for that particular shaw
00:35:33.760 so all we have to do is just pass the shot to fix up of the commit that we want to fix and
00:35:39.200 then we can pick the second commit and say that we want to fix it too um and update the implementation there as
00:35:45.760 well so what that looks like in your rebase editor if you do a git interactive rebase and you rebase off a
00:35:51.760 master uh you'll see this in your editor and if you've never seen this before it's just
00:35:58.079 you've got five commits here instead of the three because we did the commit with the dash
00:36:03.359 dash fix up command and what git does is it creates a fix up bang command or sorry a fix up bang commit
00:36:11.280 with the same subject as the um as the as the commit that you want to
00:36:17.040 fix up so in this case the way the git does this um is it associates the subjects
00:36:22.720 together um and makes it so that you can easily fix things that you've told it to fix up in
00:36:28.960 the previous slide that i was showing you where we were doing a git commit with a fix up so uh
00:36:36.400 the way the git does this is it's very simple it just makes a fix-up bang and it uses the subject of the commit that
00:36:41.839 you're trying to fix up the reason i highlight this is because you don't want to rewear it at the same time you're fixing
00:36:48.000 because if you rename the commit that you're trying to fix the fix up won't actually know where to
00:36:54.560 associate itself so try not to do that when you're doing a fix up the other thing too here is normally these fix up bank
00:37:03.119 commits will show at the very bottom of your commit history so instead of having them intermixed
00:37:08.720 like you see here because i have some advanced features turned on git wouldn't normally do that by default
00:37:14.160 you'd have to physically move those fix ups to where you want them to be i'll show you how i have this turned on
00:37:19.440 and how git did this automatically in a few more slides but just be aware that that's what i
00:37:24.560 have turned on here and then again in the second commit the reason git was able to associate
00:37:30.400 uh where the fix up needed to go with the original fix up or sorry the original commit is through the subjects both those
00:37:37.599 subjects have to match that's how git is basically doing that calculation now when we save this and we rebase and
00:37:45.040 we do a git log here we'll see that those fix ups magically got merged into the previous commits and our
00:37:51.839 fixes are in place and it looks like we just have those three commit messages from before except this time they're fixed properly
00:37:57.839 the way that we wanted to which is exactly what we wanted and a nice way to kind of
00:38:03.599 sort of massage and make those fixes and screw ups that we caught while we
00:38:08.800 were doing our feature branch work before we actually get it up for code review
00:38:14.640 so just like fix up you can also use squash it works the same way as fix up except
00:38:21.599 when you do a git rebase right it's going to halt and allow you to edit your git commit messages that's the only difference but the flow
00:38:28.400 is still the same so if we look at our git log we look at these three commits and let's say that we want to squash one
00:38:33.839 of them we just put in the shawl like we do with the fix up and then what that's going to do in your
00:38:39.440 editor is it's going to halt and it's going to say hey look uh these two commits here
00:38:46.480 uh they're gonna be squashed by the subject and in this case you can you can uh the rebase editor will halt
00:38:54.240 and then allow you to edit the commit message at the same time that it's squashing it same thing as the fix up bang except
00:39:01.359 that you get to edit the commit message in this case that's the big difference
00:39:07.119 so now that we've gone through those two techniques which are really handy for the command line the other thing i want to mention is
00:39:13.200 amend you might actually be pretty familiar with this especially if you've done a decent amount of get work
00:39:18.720 this is a pretty common tool to be aware of but i want to highlight a few things for you a man by the way always amends the
00:39:24.560 last commit and it's a great way to edit that last commit really quickly
00:39:29.680 without having to launch the entire rebase editor because that would be too much work
00:39:35.040 what you can do though is you can pass in no edit to amend your previous commit but don't
00:39:41.040 pop up the editor the reason i point this out is this is really nice for situations where you've made a change to your
00:39:47.520 implementation but you really don't need to pop up the editor you don't need to change the commit message so this is a fast way to do that
00:39:53.520 the other thing you could do is you could also push in pass in dash dash all and that'll allow you to kind of slurp
00:39:58.960 up everything you're working on and throw it into that last command granted you still want to keep your commits atomic right just implementation
00:40:05.680 plus spec if you can keep it really tight very clean but if there's a lot of changes because you're doing a refactoring or
00:40:12.240 there's a bunch of things all related this is a great way to kind of grab all that throw it into the last commit
00:40:18.640 without even popping up the editor which is a really fast workflow
00:40:24.240 uh so let's talk about a few more rebase tips a few things to actually power up and make your workflow and get
00:40:32.079 rebase workflow a lot more easier and pleasant
00:40:37.119 there's three commands that i highly highly recommend that you turn on in your global git configuration and let's walk through each one of them
00:40:43.680 the first one is abbreviate commands if you turn this on to true that'll change the get rebase editor
00:40:48.800 from typing out the long form of pick and reword and drop it'll just change them to their single
00:40:54.400 letter equivalents so instead of p i c k for
00:40:59.599 pick you just have p d for drop and r for reword i highly recommend turning this on once
00:41:05.200 you get familiar with the rebase workflow and you don't want to do all that typing auto squash by the way is how the squash
00:41:11.920 bang and the fix up bang were showing and aligned next to each other in the commit editor
00:41:18.800 git rebase editor when i was showing you in the slides previously by turning this on git will figure that
00:41:23.839 out and align those subjects together so that they're nicely they're put where you want them to be if
00:41:29.599 you leave this off as false that means you have to manually do that movement yourself
00:41:34.720 so highly recommend turning this on and the last one is auto stash auto
00:41:40.160 stash is is wonderful because normally when you rebase if you have anything dirty in your working
00:41:45.680 directory the rebase won't will halt will actually not start because it wants you
00:41:52.160 to do something with those files either commit them or delete them or stash them
00:41:57.440 so normally you would just stash them manually yourself
00:42:02.480 but what auto stash true does is before the rebase kicks off it'll detect those
00:42:07.520 changes immediately throw them into stash without even prompting you allow you to do the rebase and then when
00:42:12.800 the rebase is complete it'll pop the stash and put you back to where you're working it's a really awesome
00:42:18.480 workflow and highly recommend turning on just to let git figure all that out for you so you don't have to
00:42:23.839 manually do it yourself and by the way these are the command line arguments or the
00:42:30.400 the commands that you would type through your c your shell if you wanted to enable this globally
00:42:37.440 okay one last thing that's really powerful to configure for git rebase which is get re re-re or re-rear i i
00:42:45.440 can never pronounce it properly but it's really nice to have turned on as well so first of all you want to enable it
00:42:51.359 and what that'll do is that'll basically doing a rebase it'll any
00:42:56.640 conflicts that you resolve it'll record and stash those conflicts into like a small patch set
00:43:03.440 and um replay those anytime you do a rebase in the future so that you don't once you've resolved
00:43:09.040 the conflict resolution especially if you're rebasing heavily constantly um you're working on your feature branch
00:43:14.079 for a long time a lot of changes are coming from your your team down upstream uh as you resolve those merge
00:43:19.680 conflicts you don't want to keep having to resolve those every time you rebase it's pretty much it's kind of a pain
00:43:24.960 in those cases you can teach freebase to remember those merge conflicts and
00:43:30.880 attempt to replay those when it sees that it can actually do that automatically for you so highly highly recommend enabling that
00:43:37.680 the the last thing here is auto update if you turn this on it's off by default but if you turn
00:43:43.520 it on when you resolve those merge conflicts and you make any additional modifications to them
00:43:50.079 git will also remember that as well and try to replay that information as well so as you get better
00:43:56.000 at your rebasing and uh merge conflicts and and resolving those basically riri is a nice way in which to teach git
00:44:03.839 to also remember those changes and try to do that work for you if it detects it
00:44:09.200 and again these are the command line uh this is the command line that you want to run if you want to enable this
00:44:14.319 globally highly highly recommend turning this on it'll it'll save you some work
00:44:19.760 okay so now that we've gone through the rebase workflow we've talked about the difference between merge and rebase
00:44:25.520 how to use rebase how to configure rebase wouldn't it be awesome if you had a
00:44:31.359 coding buddy an automated little bot that could hang out with you and help ensure that your git commit messages
00:44:37.119 are well really well written and well-formed well gitland is a ruby gem that i've
00:44:42.960 developed that actually makes that possible uh what it does and we won't go into all
00:44:48.480 the details here but i'll just kind of go gloss over it briefly is it'll help you make sure that you have property
00:44:53.520 capitalized prefixes to your subjects it'll make sure that you have correct
00:44:59.280 punctuation make sure that you have a space between your subject and your body so that they're not smashed together and
00:45:04.480 they're easy to read and also make sure that you have like a nice 72 character limit so that you don't have horizontal
00:45:10.800 scrolling uh in code reviews so that they're easy to read and they're nice nicely word wrapped
00:45:16.880 and it does a ton of other things and i won't go into that all today but to install it it's very simple just
00:45:23.119 do a gem install you can check out your project switch to your feature branch and then you can
00:45:28.720 analyze it and by default get lint will analyze a feature branch it won't do anything on
00:45:34.640 your master branch because um once the code is on your master branches you can't really re rebase that or you
00:45:42.160 wouldn't want to anyway uh your team would be very upset with you if you did
00:45:47.920 but you can analyze it that way and that will tell you whether your git commits are of sound quality
00:45:53.599 there's also plenty of ability to wire it up as a get hook so that get lint is always running
00:45:59.440 every time you make a commit you can even wire up ci support on my team we actually use it as a github
00:46:05.200 action so tons of ways that you can configure get lit to actually work for both you locally
00:46:11.760 through get hooks or via your team in a ci environment
00:46:17.200 so what are the next steps right we've gone through git rebase we've talked about the flow
00:46:22.720 the configuration options and even using gitlint as a tool to help make your commits better well
00:46:28.960 the first thing you can do is uh start your workflow and just focus on it every day just take a little bit take some of the
00:46:34.880 commands i was telling you today work on them bit by bit keep getting better and better and
00:46:40.079 exercising that mental muscle until it just becomes second hand and then this will become easier
00:46:45.599 and just a natural part of your workflow without causing you any grief or pain the other thing you could
00:46:52.560 do is a lot of those commands that i was typing out and showing you it's a little bit to type right so if
00:46:57.920 you can make an alias or a function to automate that even further i highly recommend doing that and i have a link here to my dot files
00:47:03.839 project in case you want to see additional examples of how i alias things how i write functions and just make this workflow
00:47:10.640 really hum along so i don't have to do a whole lot of typing and then lastly consider adding gitland
00:47:17.280 to your projects right let it be a coding body to you but also to your team in order to make those git commit
00:47:22.960 messages tell a good compelling story so that your history is linear
00:47:28.240 debuggable and easier to read not only for your future self but also for your fellow colleagues
00:47:33.520 in case you're traveling on vacation or not even with the company anymore they'll have the joy of being able to understand what it was that you're doing
00:47:40.400 at that point in time when you're implementing that code and that's a really good feeling to leave behind as a legacy of your work
00:47:47.040 for others to enjoy and that's it that's uh my get rebased
00:47:52.720 talk kind of in a nutshell i know that's a lot to go through um there's plenty of screencast tutorials long form articles
00:47:58.960 on the alchemist collective website there highly recommend uh checking this out i'll have my slides up there as well
00:48:05.520 and all the links and everything that was in this presentation will be there for you to enjoy and dive as far as you
00:48:11.359 want to into this rabbit hole thanks so much for everyone allowing me to speak at ruby galaxy i hope you
00:48:17.839 enjoyed this and i will see you online
Explore all talks recorded at Ruby Galaxy v0.1