Gems
Like many Ruby developers, I have used Gems as long as I have been writing Ruby, but it dawned on me that I don’t really have an idea of what Gems are, so this article is to explore that.
What is a Gem and how are they written?
Gems are how we package code for reuse in other projects in the Ruby ecosystem.
The directory structure for a Gem called SomeGem
could look like this:
[user@host : ~/some_gem] $ tree
.
├── some_gem.gemspec
├── bin
│ └── ... the executables
├── lib
│ ├── some_gem
│ │ └── ... other ruby files
│ └── some_gem.rb
└── test
└── ... tests
The code for the Gem is housed in lib
, the executables under bin
, the tests in test
, and the specifications (name, version, dependencies, etc) in the gemspec
file.
If you follow a basic tutorial on creating a gem, like RubyGems.org provides, you will find that building the gem creates a file called something like SomeGem-0.0.0.gem
. This is an archive of the source code and metadata pertaining to that version of the gem. Extracting it shows a few files that can be explored
[user@host : ~/some_gem] $ file SomeGem-0.0.0.gem
SomeGem-0.0.0.gem: POSIX tar archive
[user@host : ~/some_gem] $ tar -xvf SomeGem-0.0.0.gem
x metadata.gz
x data.tar.gz
x checksums.yaml.gz
[user@host : ~/some_gem] $ gunzip metadata.gz
[user@host : ~/some_gem] $ cat metadata
... the stuff from the gemspec
[user@host : ~/some_gem] $ gunzip data.tar.gz
[user@host : ~/some_gem] $ tar -xvf data.tar
... data.tar is the archive of the source code of the gem
How do you use a gem?
After you have installed a gem to a properly configured Ruby environment (more on that later) you interact with it by requiring it in your code. require
is a method built in to Ruby but it is modified by the presence of RubyGems
, which has been packaged with Ruby for a long time. require
ing a gem means that gem’s lib
directory is added to your $LOAD_PATH
. The $LOAD_PATH
is an array of strings representing directories in the local environment.
The following will add the lib/
directory of SomeGem to the $LOAD_PATH
, making the code and classes there available.
require 'some_gem'
# => true
puts $LOAD_PATH
# => [ ..., your-ruby-install/gems/SomeGem-0.0.1/lib, ... ]
SomeGem.something
#=> something
Where does Bundler come into this?
Many of us are used to interacting with gems through the Gemfile. You can see in the above code that my example gem was installed to the directory where the current version of Ruby is installed, but in real projects we want to scope the gem to the current project. Bundler provides us with a place to keep track of the gems our project uses, where to find them, and their versions. Running bundle init
in a directory without a Gemfile adds one, so we can specify our own gems to use in this project:
[user@host : ~/bundler_example] $ bundle init
Writing new Gemfile to .../bundler_example/Gemfile
[user@host : ~/bundler_example] $ tree
.
── Gemfile
1 directory, 1 file
[user@host : ~/bundler_example] $ cat Gemfile
# frozen_string_literal: true
source "https://rubygems.org"
# gem "rails"
[user@host : ~/bundler_example] $ echo 'gem "some_gem"' >> Gemfile
[user@host : ~/bundler_example] $ cat Gemfile
# frozen_string_literal: true
source "https://rubygems.org"
# gem "rails"
gem "some_gem"