Goals for Binary Gems ("Wheels")
We want to encourage gem authors to ship precompiled binaries of their gems whenever possible (instead of the gem building an extension at install time). This is better for security – no code running at install time, and more ergonomic for users – no need to install build tools, deal with build errors, waiting for slow compilation.
Problem statement: Dealing with gems that have different implementations and requirements across different OSes, CPU architectures, libc Ruby Engines, Ruby versions, (and ruby engine versions) is challenging for both gem authors and consumers.
Challenges include:
Gem authors:
- Restricted in set of build tooling they can use
Gem consumers:
- Need to have the right build tools installed
Dependency resolution:
(Unrelated) Switch to a new /info_checksums instead of /versions
/info files will have a new metadata field, wheel=true
that indicates that the “version” (version + platform string) is in this new format
Bundler/RubyGems will resolve to the most specific match whenever possible, falling back on existing (non-wheel) gems as needed.
Enforce filename canonicalization First file in the archive is a plain-text key-value mapping of everything in the filename, so clients can easily verify that they haven’t been confused by the normalization scheme
Version strings will continue to be (version, platform)
tuples, but the platform piece will be expanded. See pypa docs on platform tags for inspiration on the different pieces. File names (and gem “full names”) will also mirror what python does for wheels.
{gem_name}-{version}-{ruby tag}-{abi tag}-{platform tag}.gem2
Given this will necessitate publishing new packages, that will only be understood by new versions of RubyGems + Bundler, I see several potential oppportunities that come along with the semi-related breaking change.
Potential opportunities:
- Package each piece of gems into their own directory:
lib/
for Ruby codeext/
for built extensionsbin/
for executablesshare/
ordata/
for data filesdoc/
for documentationsig/
for signatures- Goal is to allow installation to be as simple as copying files into the right place, permitting other implementations for installers of wheels
- Allow for non-Ruby executables to be included in the gem
- Rethink how gem specs get serialized and stored
- Can we move away from YAML that needs to reference RubyGems classes?
- Canonicalize encoding of file names in packages or allow for specification of encoding
- Obviate the need for using
__dir__
or__FILE__
in gem code, provide runtime lib like python resources module