GitHub Subtrees
- 2 minsI have been familiar with git ever since I started using Linux. I used ‘git clone’ quite a bit to compile and install latest versions of software, but that’s where most of my ‘familiarity’ was limited to. I did mess around with creating local repositories and committing but that’s all. We used SVN at work, so most of my version control knowledge is based on SVN. So, I am still figuring out lot of git stuff that most people probably already know.
So, I was looking for a way to have a single repository for a class and all my projects for that class would be stored under that repository. That’s when I came across this nifty little feature here called Git subtree merges, which allows you to manage multiple projects under a single repository.
The workflow as I understand it is as below:
-  
I created a repository called
advanced-data-structures. This is my main repository. -  
Then I created another repository called
RedBlackEventCounter. This holds the actual project code. Made changes, committed and pushed to this remote. -  
Then I added the remote of
RedBlackEventCounterto the master ofadvanced-data-structures.git remote add -f RedBlackEventCounter https://github.com/sayakbiswas/RedBlackEventCounter.git -  
Merged
RedBlackEventCounterintoadvanced-data-structures.git merge -s ours --no-commit RedBlackEventCounter/master -  
Copied the git data of
RedBlackEventCounterrepository into a new directory in theadvanced-data-structuresrepository.git read-tree --prefix=RedBlackEventCounter/ -u RedBlackEventCounter/master -  
Commit and push the changes.
git commit -m "Added RedBlackEventCounter reference into advanced-data-structures" git push origin master 
Now, this subtree doesn’t automatically sync with the changes made in the upstream repository. GitHub’s documentation recommends using the below command to update:
	git pull -s subtree remotename branchname
[Update - 03/15/2016]: The above command seems to work properly now. I’m not sure what had happened earlier; probably some mistake on my part. Need to dig deeper. So my workflow is now:
	git pull -s subtree RedBlackEventCounter master
	git push origin master
The below is not needed anymore.
But for some reason, it doesn’t work out for me. I keep getting the error
	error: Entry %filename% overlaps with %filename%. Cannot bind.
So, what I do is create a temporary directory and pull in everything from subtree remote. Copy over everything from this directory to the subtree directory. Remove the temporary directory, stage all, commit and push.
	mkdir RedBlackTemp
	git read-tree --prefix=RedBlackTemp/ -u RedBlackEventCounter/master
	git commit -m "Pulled upstream changes"
	cp -rf RedBlackTemp/* RedBlackEventCounter/
	git rm -rf RedBlackTemp
	git add .
	git commit -m "Merged upstream changes into master"
	git push origin master
I know it is quite cumbersome to do this everytime you make a change. I guess there is a better way that I just don’t know yet. So, if anybody has an idea, please let me know.