This article was written for the ruby advent 2008, day 20. I’m reposting it here so I have a backup under my own roof.
If you already know about vendoring and why it’s good for you, feel free to skip directly directly to the braid example#.
If you still need convincing, read on.
It’s a well-known fact that every piece of code that you write should be version controlled.
Not reinventing the wheel is also a good idea that will keep you happy. In practice, not reinventing the wheel means trying to use libraries or plugins written by other happy open source developers. This means that you now have external dependencies.
Generally, you want your project to be as stable as possible. However, you will also want to update your external dependencies quite often to make use of their new and shiny features.
The simplest way is to just make a note in the documentation and hope that all our users and contributors will setup things correctly. But this is an error prone procedure and it might be too much work for some to the extent that they might not bother.
Still, if all that you require from your external dependencies is an official release, most ruby libraries are released as a gem so, you could just use rubygems and be done with it.
Using rubygems is all fine until you need to make some custom changes to a library. These might be some changes you are not ready to submit to the original projects, or even changes that you submitted but have not yet been integrated. Or, maybe they don’t even make sense for anyone else but you.
Due to ruby’s nature, in some cases, you can avoid having all of the external code in your repository with some clever monkey-patching. But, if you have changes that are not trivial, sooner or later, you will need to bundle the external dependency with your project.
A simple and quick idea is to just take all the source code of your dependency form it’s repository, copy and dump it in a directory, remove the .git or .svn directories and simply commit it. This is a good enough starting point and it’s exactly what script/plugin does in rails. You are now free to tweak it to your heart’s content and then commit it to your own project’s repository.
Let’s say that you now vendored a few libraries and happily hacked on them and your project for a while. Suddenly, a security flaw is discovered in one of the external libraries. It’s authors are quick to provide a fix and you are keen to have it.
So, if you used the “copy and dump” approach, you’re kinda screwed, since you need both the security fix and your magic tweaks. You have to either manually merge the changes, discard your changes and start from zero with the latest version or, ignore the security issue altogether. I’m not sure which one is scarier :).
Let’s just assume that you definitely need both the security fix and all your custom changes. Basically, to continue, you have two options. You could either apply the new fixes on top of your code or you could reapply your fixes on top of the new library code.
- Applying new fixes on top of your code
- Applying your custom changes on top of the most recent version
Besides needing major custom changes, there are two other major reasons you might want to bundle your external dependencies.
One is that if you are using a new library that’s in active development, you will most likely need to update with bugfixes quite often.
The other is that if you are developing a web application that needs to be deployed, you will not want to depend on external repositories being down and preventing your deploy.
So, most of the time you will want to bundle external dependencies anyway, regardless of any custom changes you may or may not have.
If you made it this far, you should have a pretty good picture about your vendoring needs. I’ll just make this easy to skim by recapping things.
You don’t need vendoring if:
- You are writing a library that depends on another library. Just use rubygems. Examples of this are most gems like capistrano, braid, haml, etc.
- You are writing a library that depends on another library and needs minor tweaks. Just use rubygems and a small monkey-patch.
- You are writing an application that will be deployed and you don’t care if you can’t deploy :). Example: websites that fail to deploy.
You need vendoring if:
- You need close integration with an external library. Example: gitnub uses grit internally. So, it’s bundled.
- You need to update to new releases of the external library relatively often. Example: all rails applications that use plugins such as will_paginate, acts_as_list, etc
- You need to be able to deploy even if everything is down, except the production server :). If have vendored everything, you can still deploy form your local directory.
- You want other people working on the project to be able to quickly get started
So, basically, you need vendoring most of the time. There are several tools out there to help with this. Braid is the first one that was written especially to take advantage of git’s power. Keep reading to see how easy braid makes things for you.
Using braid to manage dependencies in git
Let’s go through a simple scenario to see how easy braid makes dependency management by using a completely silly example. You will notice that braid commands are very simple and scarcely used. To make them stand out, they will be highlighted.
The requirements are simple: we need to make an app that makes it clear that the hoff should not be hassled. Since this is a simple app, let’s use sinatra.
Okay, let’s start by initializing an empty git repository:$ mkdir donthasslethehoff $ cd donthasslethehoff $ git init Initialized empty Git repository in /Users/chelu/tmp/donthasslethehoff/.git/ $ echo donthasslethehoff > README $ git add README $ git commit -m “initial commit” Created initial commit c6eeba4: initial commit 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 README
To start, we’ll vendor sinatra using braid:$ braid add git://github.com/bmizerany/sinatra.git
This will simply create a normal commit in your repository with all that is needed to make braid work. If someone clones your repository, they don’t have to do anything. They don’t need braid installed and don’t even need to know you’re using braid.
You will now have sinatra in a directory inside your app. Let’s add the code to display the message.$ cat > hoff.rb $:.unshift File.dirname(FILE) + ‘/sinatra/lib’ require ‘sinatra’ get ‘/’ do “don’t hassle the hoff” end ^D
And, we’re done. Let’s try it out:$ ruby hoff.rb
Load up http://localhost:4567/ in a browser, and see that it worked. All ok? Go ahead and commit our changes:$ git add . $ git commit -m “not hassling the hoff”
Now, the site looks good but, our app must make it very clear that the hoff should not be hassled! So, let’s go ahead and change sinatra’s console messages to be more clear about this. This is one of those have to tweak a dependency situations I mentioned earlier.
So, go to sinatra/lib/sinatra.rb and search for the line that says: “has taken the stage”. Add a new line just below it:puts “Sinatra says: Remember folks. Don’t hassle the Hoff!”
Now we’re talking. Everytime you start your sinatra app, you will be reminded not to hassle he hoff. Restart to see the new message and commit the change:$ git add . $ git commit -m “not hassling the hoff hack”
You can now also use the braid diff command to see all of your custom changes to sinatra:$ braid diff sinatra
After a while, you will want to use braid to update your sinatra mirror to get bugfixes and new features. Just make sure you don’t have any uncommitted local changes and run:$ braid update sinatra
This will bring you sinatra goodies and still leave your modifications in place. Under the hood, braid will do some tricks so that the power of git is leveraged to merge the changes in smoothly. Again, the end result will be just a normal git commit.
I hope this made it clear how easy and simple using braid is. You can find more examples on the braid wiki.
You can file bugs and feature requests at the braid project on lighthouse. And there is also a google group for discussions and community support: braid-gem google group.
Braid is hosted on github at http://github.com/evilchelu/braid/tree/master and we are always glad to receive patches and suggestions.
Braid is a simple tool to help track git and svn vendor branches in a git repository.
This is a major release that brings two main features (no more braid/track branch and better merging support) and lots of usability changes, bugfixes and tweaks.
The braid project page has now moved to github, mainly because we wanted to use the wiki for documentation.
Braid now also has a google group. If you have questions or ideas about braid, please subscribe to http://groups.google.com/group/braid-gem.
Installinggem install braid
Important changes from 0.4
- removed braid/track branch
- use merge-recursive to allow changes from upstream to be automatically merged in
- added local cache to workaround git fetch limitations
Other changes from 0.4
- improved messages
- refactoring all around
- moved to test/spec from rspec
- basic integration tests
Coming soon in braid 0.6
- braid switch
- braid format-patch
- braid import
- braid outdated
- braid list
- bash completion
- possibly, windows compatibility
Braid is a simple tool to help track git and svn vendor branches in a git repository.
This release fixes a bunch of bugs and adds important missing features which allow braid to actually be useful to people working in teams and it also stops polluting the git history with messages from the imported mirrors.
This release is possible because of the work of Norbert Crombach, who totally rocks!
- all new mirrors are created squashed, which means that you’ll only get one commit when adding a mirror
- if you already have braid mirrors you need to update to squashed format.
- braid now can be used by all developers on a project. There’s a guide for this on the wiki.
- braid diff
- braid update can now lock to a revision
- commands are only executed if there are no local changes and the git version is above 126.96.36.199
- bunch of bugfixes and cleanups
Install with rubygems
Unfortunately the github gem server is not working for everyone yet so please install from the git repository if this doesn’t workgem sources -a http://gems.github.com/ # only need to do this once gem install evilchelu-braid
Get it from the git repository
Get a clone of the git repository using:git clone git://github.com/evilchelu/braid.git cd braid rake install_gem braid —help # see usage
Open source works!
This release is very special for me because it showed me, in a personal way, that open source really works.
I was about to give up on braid at the end of March mainly because I had no idea on how to fix the history polluting and being able to work in teams. And there were also bugs and the code looked a bit ugly.
Fortunately, I didn’t shut it down because after about a month where I didn’t do any updates to braid, Norbert Crombach (github) told me he forked the project on github and fixed all the major outstanding issues! I was shocked and awed by his effort and, since all the major stuff is now fixed, I found new motivation to work on braid again. Thanks, dude! :)
We’ve been tweaking and cleaning up things for the last week to get this handy release out. We have ideas for a number of other features for the next releases so, head over to the braid project page on github and watch it to get updates. Github is great and it makes patching and merging so easy that we hope you’ll consider forking the project and contributing with code :).
If you got braid from the git repository before 12:40GMT, please pull the latest changes.
The 0.3.0 version has a bug where it will clone the mirrored svn repositories entierly. braid 0.3.1 just grabs the HEAD revision and this is significantly faster.
- braid was fetching entire repositories for svn mirros. changed to fetch only HEAD
- workaround for stupid git svn behaviour. it is transforming all _ (underscores) into . (periods) in git svn remote branch names
- braid now is replacing all _ with – preemtively
- braid only works with a quite recent git (I’m using latest from the git git repo). I know that git 1.5.2 definitely doesn’t work (git fetch is acting up). Not sure about versions in between.
Updatingcd braid git pull rake install_gem
Get itgit clone git://github.com/evilchelu/braid.git cd braid rake install_gem braid —help # see usage
Mihai Anca has written a nice article showing some real world braid usage: Installing rspec with braid
He also helped track down the stupid bugs in the 0.3.0 release together with Dominic Owens, who is leading the bestforexbrokersuk.com project. I have beers already prepared for them.
See all posts in the archives