Yeah learned this the hard way.
- The main reason I use git is it allows me to make mistakes without hard consequences. Any fuckup is just one reset away from being fixed. I like to: I have to fix this thing. While attempting to fix it I discover there is another thing that needs fixing on which everything revolves. I fix the second thing and commit. I’m now free to fuck around the code all I want and I’m sure I won’t lose that fix. - For this I really like to use --fixup when I find out the change was not completely right or does not fit well with some other changes I need to do. I really like git absorb which automates this a bit. 
- I’ve had juniors who didn’t believe this, so just to say it: If you know what you’re doing, practically any Git problem is recoverable. - The one major exception is if you delete your local changes before committing them. - Yeah.But many of them are extremely annoying. Specifically screwing up rebase. It is recoverable, but very annoying. - That said I have seen juniors make two other common mistakes. - Pushing your commit without fetching
- Continuing on a branch even after it was merged.
 - I’m fed up with these two. Yesterday I had to cherry-pick to solve a combination of these two. - Maybe I’m just a wizard, or I don’t know what y’all are talking about, but rebases aren’t special. If you use - git reflogit just tells you where you used to be before the rebase. You don’t have to fix anything, git is append only. See where the rebase started in reflog, it’ll say rebase in the log line, then- git reset --hard THAT_HASH- Pushing without fetching should be an error. So either they got the error, didn’t think about it, and then force pushed, or someone taught them to just always force push. In either case the problem is the force part, the tool is built to prevent this by default. - Continuing after merge should be pretty easy? I’d assume rebase just does it? Unless the merge was a squash merge or rebase merge. Then yeah, slightly annoying, but still mostly - git rebase -iand then delete lines look like they were already merged?- See all this is fine for someone with good experience in git. They know how to solve the screw up. But wih junior devs, who don’t know much about it, they will just get confused and stuck. And one of the senior has to involve and help them solve. This is just annoying because these can be avoided very easily. Until they understand the pattern of how everyone operates with git, it just creates issues. - To me first time causing this issue is completely fine. I will personally sit with them and explain then what went wrong and how to recover. Most of them will repeat it again, act clueless and talk like they are seeing this for the first time in their life. That is the difficult part to me. - May be I’m just old school, and a grumpy old person, even though I’m not that aged. - Oh, the human interaction is annoying! Yeah gotcha. That makes more sense! 
- Sounds like the onboarding process needs to have a step in it that says “here’s a link to a git tutorial, read this and get familiar with using git, as it’s an integral tool that you will use every single day on the job”. Bonus points for providing a sample repo that juniors can use to mess around with git, extra bonus points for including steps in the onboarding materials for the juniors to set up their own repos to play around with. 
 
- Same… My usual strategy: rebase, if conflict abort and merge, if no conflict continue; merge always with explicit commits to master / main (no fucking squashing); keep task references in branch names and commit messages. - Same, but typically I will just resolve the conflicts during the rebase. Makes for cleaner commit history. Merge commits are for combining multiple big unrelated pieces of work together, where rebasing would be too annoying (let’s say 100s of commits each). - In my cases I has to solve same code conflicts multiple times during a rebase, so I just don’t try them when hit with conflicts. - I fail to see the benefits of “clean” git history - In my cases I has to solve same code conflicts multiple times during a rebase, so I just don’t try them when hit with conflicts. - Yeah if you have two branches, both with a bunch of commits which all modify the same areas of code, and reordering the commits doesn’t help, I can see how it is easier to merge. - I fail to see the benefits of “clean” git history - Well, if the commit history is clean and mostly linear, it’s much easier to read, understand and review. - git blamewill also be much nicer which is really important for debugging big codebases. Of course it’s a tradeoff, as usual.- Maybe I just haven’t been exposed to bad examples. Never had any issues with blame and merge commits. 
 
 
 
 
- Continuing on a squash merged branch is very annoying, and I had to deal with this in one repo regularly… Luckily I was annoyed enough to research about this and found out about - rebase --onto "main merge commit" "branch merged commit".
 
- Is it ok to continue on a branch if you also merge back main into it? Like, branch gets merged into main on remote, local main pull, local merge main into local branch, push branch? - So this workflow is needed if you are working on a public, i.e. multiple devs collaborating on a single branch, scenario. But it is much better to avoid this as much as possible. Usually it is a ‘scoping’ issue, where you create a branch that is too broad. For example ‘api-for-frontend’, which is a massive thing. - But let us say you absolutely have to get multiple devs on same branch, then this workflow is totally fine. There is nothing wrong in it. - In our org we prefer to delete the branch after merge. In a way it says ‘this branch is closed’. This is to encourage devs to define smaller and more logically scoped branches. - I want to take this opportunity to say that, branch is just a label on a commit, with some additional functions. Once you start focus on commits and lineage of the commits, then branches become some what irrelevant. - Aha. I was part of a project where each dev had their own long running branch for non-specific work and this was the norm, but it always felt clunky. And often resulted in merge issues. - That is a very weird setup. I have no clue why that flow is needed in the first place. Branches should be something disposable easily. What was the logic behind the setup? Any idea? - Oh I know the reason, nobody knew git and had just worked alone before. 
 
 
 
- Is it ok to continue on a branch if you also merge back main into it? - On some repositories, sure. - But better maintained repositories don’t allow merge commits (because merge commits suck), and so will have squashed (or rebased) on merge. - (If squashed) The squash will have changed commit IDs, so a long running branch rebased won’t benefit from a clean shared commit history. - So it can work, but “you’re gonna have a bad time.” - In general, git works best if branches are thrown away as soon and as often as possible. - (Edit: Good clarification in response below, added here for consistency and accuracy.) - You don’t have to squash to avoid merge commits. Instead, you can - git rebase mainto update your branch. Effectively, this will rewrite the history of your branch, as if you had just branched from the main-branch and then instantly coded all your changes on top of that. (Well, the commit timestamps won’t change, but they will sit on top of the changes of the main-branch.)- Afterwards, you should be able to merge into - mainby switching to it and then running- git merge --ff-only your_branch.
 Because all the changes sit on top of the main-branch commits, it should be able to fast-forward. No actual merging needs to take place then. You’ve already resolved any conflicts while rebasing.- This also allows you to keep branches for longer, so long as you frequently rebase and merge back. 
 
 
- Specifically screwing up rebase. It is recoverable, but very annoying. - WDYM? Typically - git rebase --abortwill “just work”. If you have specifically done something really wrong, just look into- git reflogand then- git reset --hardto whatever commit you were before the rebase.- Pushing your commit without fetching - Git won’t allow you to do that if you set up your server properly. It will force you to pull first. I have - [pull] rebase = truein my settings so that it always rebases my commits instead of merging them, which makes for much cleaner history.- Continuing on a branch even after it was merged. - This generally shouldn’t be a problem, you can just rebase the branch afterwards and it will be fine (the common commits will typically just be dropped). - The problem is not when I have to rebase. I know how to handle it. But with juniors they approach us only when things are in a really bad situation, where they cluelessly applied some commands they found on internet or from an LLM. Then it is very annoying to sit down and untangle the mess they created. - And regarding the pushing without fetching, it is usually a different branch. So they won’t incorporate the new changes in the main branch into their working branch, but just push their work into a branch. Again not a big deal. Just annoying. 
 
- I’ll get chastised, but when I screw up a rebase I create a backup branch and then have an LLM do the fixing… - I’m not proud of it, but I’m a lazy man who doesn’t want to resolve the same conflict 32 times. - Having to resolve the same conflict multiple times suggests excess noise in your git history. You might want to pay closer attention to creating a useful git history. It’ll help with any future archaeology, and it’ll also help rebasing go smoothly. 
 
 
- All hail the mighty reflog! - Saviour of us all! - o7 
 
- IntelliJ has local history for that case. 
- Does deleting the repo off disk and redownloading from remote considered recovering? If so I’m a git expert. - Delete? Never. - mv git_repo git_repo.bad- I do this sometimes but then I forget about it and then node_modules in each repo fills my SSD. - I mostly only use git at work and I guess I’ve never had the same work machine long enough to worry about this. It helps that each of our repos is fairly small. At least the ones I touch. 
 
 
 
- I’ve even recovered from deleting local changes I forgot how but there is a way! (It’s not something to rely on but can help in a pinch!) - Well, if you did commit it, but just hadn’t pushed it yet, and then somehow lost that commit, then - git reflogwould be the tool for it.
 Without a commit, sometimes you may have already staged some changes for an upcoming commit and can roll back to that.- But if neither of those are the case, then I can’t really imagine how Git should help you there. You haven’t told Git about those changes yet, so it’s out of scope. 
 At that point, you better hope your editor’s undo history goes back far enough…- This might have been what it was as you mentioned git reflog. I don’t remember clearly since I’m usually pretty good at not making this mistake. 
 
 
- Fun fact! If you added a file but never committed it you can still recover it with - git fsck --lost-and-found. The changes were still added to the git object store, and by grepping through the dangling blobs you can find the object containing the changes you care about.
 
- git reflog - git re-flog is what you do with those idiots who mess up the repo so that someone else has to come in and fix it again. 
- Reflog, when flogging isn’t enough. 
- Fuck, you beat me to it - Rebase the comment chain and force push! - The number of times I get the compulsion to perform git operations in daily life is disturbing - You’re seeing the matrix. 
 
 
 
- I hear people praise - git refloga lot, but when do y’all use this? Most I’ve ever needed is- git log --graphfor history checking. Maybe if I was in a situation where I knew the code on local branch X was working on Friday, but not anymore on Monday, then I could use- git reflogto go back to Friday for that local branch right? Is that the idea?- Power users rebase with squashes and fixups multiple times a day. Especially if the job’s integration process isn’t enforcing long living branches. - Reflog is useful then, because you literally rewrite history every rebase. 
- It’s not often, but a bad git reset, or a messed up rebase of that branch that’s 8 months old that you’ve picked up again. 
- I only use it when I’ve royally messed up and the commit I need to get back is no longer referenced anywhere. Accidentally deleted a branch, finished a merge or rebase before realizing I messed up, that kind of thing, just use the reflog to find it again, get a branch pointing to it, then try again. 
- git logwill only show you commits in your history. If you’re only ever working forwards, this will contain all the stuff you’ll ever need.- But if you’re rewriting history, like with a rebase or squash or something, or you’re deleting branches without merging them, then you can run into a situation where the official history of your branch doesn’t contain some of the commits that used to exist, and in fact still exist but are unlinked from anywhere. So reflog is the log of where you’ve been, even if where you’ve been isn’t in the official history anymore, so you can find your way back to previous states even if there isn’t otherwise a name for them. - If all you care about is your current history, git can use the dates of commits just fine to see where you were on Thursday without needing the reflog. 
- When I’m not using - git reflog, it is only because I a’m not making big enough- gitscrew ups.
- It’s mostly for undoing a rebase or merge you’re not happy with. 
- I really only need it when I done fucked up. 
 
 
- Use git for any code you write. Yes, even a simple script.
- Commit and push often. More often than you think is reasonable. You can always rebase / fixup / squash / edit but you can’t recover what you didn’t commit.
- ???
- Profit.
 - Seriously, once you commited something to the repo it’s hard to lose it. Unless you delete - .git. But a this point frequent pushing has your back.- I know git can be hard to grasp in the beginning. It was hard for me too. I highly encourage everyone to put in the effort to understand it. But if you don’t want to do that right now just use it. Just commit and push. It will pay off. - (3) Get annoyed by constantly increasing Code Coverage Requirements on untestable (often legacy) code. Even afding comments requires code coverage go up! Line must always go up!
- Change step 2 to “Commit and push ONLY when absolutely necessary. Because adding comments often requires a rewrite of untestable code.”
- Go back to Step 2 and wait for a major bug.
 - Why would you care about code coverage requirements on a branch that is currently in development? My work in progress commits might not build because they don’t even compile, let alone have passing tests. The only time I care about the build passing and meeting requirements is when I’m ready to create the pull request. 
- Our company “architects” decided we needed 80% coverage minimum. On a mostly .Net 4.8 codebase. Hundreds of programs used in prod, with barely any tests in sight. 
 It’s a fucking nightmare.- Ah, the classical “just introduce tests in a legacy codebase”, what can go wrong? - My condolences, it’s always a BITCH to handle - Fixing that bug you just found in prod : half a day. Adding enough tests so the app is now at 80% coverage : a whole fucking week. My student colleague was not impressed. I was like “yup, that’s what jobs are like”. 
 
 
- 3 is not related to using git in any way. I’m not really sure what you mean in 4. I didn’t mean making a lot of changes, I meant that you should not wait with committing until you have a finished feature / fix / whatever. Commit after each refactor, commit after adding a new testable unit. It’s always better to have more checkpoints. If your team does code review, they will appreciate atomic commits too. 
 
 
 
- Git repository operations are (almost?) always recoverable. - git reflogis your friend.- The filesystem operations are another story. Handle with care. - Wait, what does reflog do? - Think of it like your browser history but for Git. It’s a list of the SHAs related to your recent operations. - And because Git is a content-addressable data store, a SHA is basically like a URL. Even if a branch no longer exists, if you know the SHA it pointed to then you can still check out the exact contents of that branch. The reflog helps you find that. - Goddamit… It’s ref log, not re flog. I thought this was related to blame and never touched it lmao  - I guess “git sanasaryan-han-torture” was an overkill sometimes… 
- Re flog is when your devs don’t follow your guides and mess uo git history again. - I legit thought it was to publicly call out a user to be (metaphorically) flogged - And while it’s not how I do things, I totally understand the impulse. Tell me that doesn’t sound like something that could exist - Knowing that - git blameexists, and Linus in general, indeed a- git flogisn’t that improbable.
 
 
 
 
 
 
- Special shout out to the person who committed a gigabyte memory dump a few years ago. Even with a shallow clone, it’s pretty darn slow now. - We can’t rewrite history to remove it since other things rely on the commit IDs not changing. - Oh well. - Sounds like a flawed workflow, if this didn’t go through at least code review. Was it committed directly to master? - Curious to know what kind of system relies on hashed not changing? Technically the hashes don’t change, but a new set of commits is made. The history diverges, and you can still keep the old master if you need it for some time, even cherry pick patches to it… 
- The guy at work who managed git before me, well didn’t quite have the knowledge I do and was not using LFS. In one of the main repos a 200mb binary was pushed 80+ times. This is not the only file that this happened to. Even if you do a shallow clone, you still need to add the commit depth eventually. It’s a nightmare. 
- Special shout out to the person who committed a gigabyte memory dump a few years ago. Even with a shallow clone, it’s pretty darn slow now. - If you want to be the team’s hero, I’ve had good luck removing old commits using - git filter repo.- https://hibbard.eu/erase-sensitive-files-git-history-filter-repo/ - We can’t rewrite history to remove it since other things rely on the commit IDs not changing. - Oh right. Oof. - I would be working to arrange an accident for those other things. Those other things probably need to be retired. 
- Why can’t you just remove the file? - Because the git history will still keep it completely to be able to restore it once you go to a commit in which it wasn’t deleted. - And you cannot tell git to completely forget about it as that requires a rewrite of history and this changes all commit hashes which are touched (and any future ones). 
- They most likely did in a later commit. However the commit adding it can not be removed as OP said. So in order for git to be able to work properly the file is still in the repository in case someone wants to check out a commit where it still was present. 
 
- why did you merge it? 
 
- You know… A version control system… That class of software that makes it possible for you to recover from any error you commit. - yeah, the whole thread is junior af 
 
- With Jujutsu (which is compatible with git), you can just - jj undo- Been using it for over a year now and not being scared of trying operations is such a boon. It helps so much with learning when you know you can just roll back to an earlier state. - I’ve had zero issues with it so far and no one at work noticed anything different, other than there being a bit more rebase spam on PRs. 
- That is so cool. Why doesn’t git have this already? - I mean, by definition, it does. It just involves parsing through the git log and a bunch of unintuitive, archaic commands. 
- idk but my dotfiles do: - alias undo="git reset --soft HEAD~1"
 
- Can’t ya just git reset? - Anything you can do in Jujitsu you can do in git… The big difference is a paradime change: - -instead of a working directory that has pending changes you need to add than commit, all changes are in a commit that is lacking metadata. - The system has better “editing” of local history to set that meta data. But once you push to a shared repo you run the usual risks of force pushing. - I’m not sold, rather git not do anything until asked and just run git status constantly but I don’t have first hand experience… I would theory it would be more likely to add a file you didn’t mean to… Unlike those who use windows guis for git and forget to add new files. 
 
 
- Isn’t it the exact opposite? - I learned that you can never make a mistake if you aren’t using git, or any other way for having access to old versions. - With git it is really easy to get back to an old version, or bisect commits to figure out what exact change was the mistake. - The only way I understand this joke is more about not wanting to be caught making a mistake, because that is pretty easy. In other methods figuring out who did the mistake might be impossible. - This is not about mistakes in the Git-managed code. This is about mistakes in the Git commands themselves. Anything that involves merging/rebasing/conflict resolution can potentially be botched. These mistakes are usually fixable, but: - Fixing it requires some Git proficiency behind the level of the common Git user.
- If you don’t catch it in time, and only find the mistake when it’s deep in your layers of Git history - well, good luck.
 - Went to tech elevator boot camp, was a decent experience even if I don’t find myself doing exactly what I was expecting to do. Life is that way though. - Anyways, my first week I fucked some git stuff up so bad I became the go to guy when anyone had any git issues because I had to learn so much to undo my egregious error. I don’t remember now exactly what it was but it took some serious git magic to sort. - Just saying that point 1 is very true. And yeah don’t make mistakes in git. 
- Like accidentally committing multi-GB files and years down some poor mf (me in this case) has to figure out how to clear the files from history because the server is running out of space. 
- Short of manually deleting .git you can always find any commit, you can walk backwards through your reference lof if it comes to it, the only real risk is throwing out unstaged changes. 
- Hmm… I am using git for maybe 15 years… Maybe I’m just too familiar with it… and have forgotten my initial struggles… To me using git comes natural… And I normally pay a lot of attention to every single commit, since I started working on patches for the Linux kernel. I often rebase and reorder commits many times, before pushing/merging them into a branch where continuity matters. 
 
- You’re confusing errors in your code, and errors while doing some git command you thought you knew. Lucky you, it’s clearly never happened to you because you don’t mess around with it. - Sure, I sometimes messed up with git, but a git reset , checkout, rebase or filter-branch (In the extreme cases) normally fixes it, but real issues are very rare. And I use git a lot… But only the CLI, maybe people have issues with GUIs?  - Funny those are commands I avoid… They all have to do with editing history which I know there is a vocal group here that loves “clean” history but that isn’t what happened. - sure merge full features so you can roll back a feature… And if something is really off I might start from a snapshot commit and cherry pick/merge a bunch in but usually history is histoy… If submitting to a public project I may make a new branch with the cleaned version but why edit in line. That is risking issues. 
- 👌 
 
 
 
 
- As long as you never touch the rebase button, you’ll be fine. Probably. - Don’t be afraid of rebases, they are an essential tool in Git. - This particular fear can only be addressed by understanding. - I don’t understand it. Every time I see something about a rebase it’s some purist telling me it’s “cleaner”. Never got it to do what it says on the tin, and never hit a situation that I couldn’t solve using more straightforward tools like merge. - What’s your mental model for a Git commit, and a Git branch? - Once I properly learned those two concepts, understanding rebases became a lot easier. - I’ll try to explain it to the best of my abilities. - Think of a commit being a patch - a description of how to take a particular file from one state to another
- A branch is a list of patches to be applied in order, from the point where the branch was created until the last commit on the branch
 - When you rebase a particular branch, what you’re essentially doing is taking all of the commits that are currently on your branch, checking out the other branch, and then applying each of the patches in order on that new branch. - A rebase can be cleanly applied if the premise for each commit has not changed when applied, but if the premise has changed, you get a conflict to be resolved before being able to continue the rebase. - I mentally model a rebase a bit as a fast version of how it would look like to build the branch I was on, but on top of the branch I’m rebasing on. - That’s a good explanation of what it’s supposed to do. That was how I understood it as well. - But anytime I’ve tried it, I’ve ended up with conflicts where there shouldn’t be (like, I already solved that conflict when I merged earlier) and/or completely undesirable results in the end (for instance, some of my changes are just NOT in the result). - So I just gave up on the whole feature. Simpler to just merge the source branch into mine. - Depending on how structured your commits have been, it can either be very difficult to get a rebase through or a complete breeze. There are some features to make it easier - - rererebeing the main one I’m thinking about.- Is that what interactive rebase tools use? - I don’t do CLI git - You enable it using - git config, after that it will apply to whatever frontend you’re using.
 
 
 
 
 
 
- …button 
- … and force push. - If you ever find yourself in a situation where rebase or a force push seems to be the solution, take a step back, clone your repo in a new directory and copy the changes into you’re new checkout - ‘cause you gon’ and screwed somethin’ up, son. - I rebase and force push daily. I like squashing all my commits, and our main branch moves quickly so I rebase off that often. Zero issues for me. - Same. And even if you were to fuck up, have people never heard of the - reflog…?- Every job I’ve worked at it’s been the expectation to regularly rebase your feature branch on main, to squash your commits (and then force push, obv), and for most projects to do rebase-merges of PRs rather than creating merge commits. Even the, uh, less gifted developers never had an issue with this. - I think people just hear the meme about git being hard somewhere and then use that as an excuse to never learn. - Why would you want to squash? Feels weird to willingly give up history granularity… - Because a commit should be an “indivisible” unit, in the sense that “should this be a separate commit?” equates to “would I ever want to revert just these changes?”. - IDK about your commit histories, but if I’d leave everything in there, there’d be a ton of fixup commits just fixing spelling, satisfying the linter,… - Also, changes requested by reviewers: those fixups almost always belong to the same commit, it makes no sense for them to be separate. - And finally, I guess you do technically give up some granularity, but you gain an immense amount of readability of your commit history. - Huh. Never thought of it that way. I was never bothered by a long commit history at all. Search and filter tools in the git client always get me where I want. - The one issue I have is when there are way too many extant branches and the graph takes up happy half my screen. - But that’s more of a Fork issue than it is a fundamental one. The Fork dev could conceivably find a solution for that. - Either way, I guess I see what you mean. I’m just not that strict about commits. Commits just for the linter aren’t a thing since we have a pre-commit hook for that, and typo-fixing commits… Well, they happen, but they’re typically not numerous enough that I’d find them to be any sort of issue. - As for whether I’d really want to revert a particular change – while I work, yes. Afterwards, I see what you mean; i could probably squash 50 commits into 15 or something. But when I think about the time investment of reviewing every commit and thinking about how they ought to be grouped together before making my merge request… I have a lot of trouble convincing myself it’s a good time investment. - Maybe I’d think otherwise if we had a huge team. We have maybe 10 devs on this project at any given time. 
 
 
 
- Yeah, I hate it when my repo is a chain of merge commits. I want to see actual changes to the code, not branch management history. - I’m the opposite. I just let git take care of the stupid content. Why mess with the commit graph? Merging locally (instead of squashing) works better with merge requests because the graph clearly shows what changes went where. - I do some branch maintenance on my local branch (rebasing) until there are conflicts, but other than that I don’t see any benefit for messing with commit history. 
 
 
- I rebase and force push PR branches all the time. Master is moving quicker than my PR. - Yeah, our whole workflow is based on rebasing our feature branches on develop. Makes for a clean git log. :) - Don’t be afraid of - git reset --hardif you rebased with the button on GitHub/gitlab, though. :D
 
- Force pushing is necessary when using rebases, and rebases are an essential tool, so you should not be afraid to force push under normal circumstances. 
- A few days ago I had to gently explain to someone why their rebase-and-force-push strategy not only prevented the use of “review latest” feature on GitHub, but was also pointless because all PRs are squash committed to main. - They didn’t get it and now they seem a little mad at me. - I’m guessing this is in reference to a scenario where a review of the PR has already been performed, and the rebase+force push is made to introduce new changes to the PR, possibly to address PR feedback. - I agree that these changes should be made in separate commits, for the benefit of the reviewer. - There are other scenarios where rebases are appropriate though, such as getting potentially incompatible changes from the main branch into the PR, and here I believe a rebase+force push is the right tool for the job. - Oh there’s totally a time and place for rebase strategies, this just wasn’t one of them. - Git’s biggest problems come from 
 people taking ritualistic views on what is “right” instead of thinking about which strategies work best for the situation, project, and team.
 
 
- Hmm, I’m less afraid of force push. It does what it says on the tin. If I pushed a fuck-uo to remote and a reset is the simplest way out, you can bet I’m force-pushing that reset. 
 
- Even if you rebase you can still recover the original commits until they are garbage collected. You are generally safe as long as the - .gitdirectory isn’t deleted, in which case your whole history is gone anyway.
 
- I don’t think you should be scared of git. - Unless by “scared” you mean “I pasted from ChatGPT and now I don’t know what to do,” in which case, please be more scared. 
- Skill issue 
 - Did they have a hose behind his head? - It’s sweat 
 
 
- We need version control for the version control. - git gud, son 
 
- git reflog is your friend - The worst thing you could do is delete your local git repo. 

















