/backports

The latest features of Ruby backported to older versions.

Primary LanguageRubyMIT LicenseMIT

  • Yearning to use some of the new cool features in Ruby 2.0.0 while using 1.8.6?

  • One of your client is stuck with Ruby 1.8.6 but you want to use a gem using some features of 1.8.7?

  • Can’t remember if you can use Array#sample or String#each_char on a friend’s box?

This gem is for you!

The goal of ‘backports’ is to make it easier to write ruby code that runs across different versions of Ruby.

For example, if you want to use flat_map, even in Ruby implementations that don’t include it, as well as the new bsearch method:

require 'backports/1.9.2/enumerable/flat_map'
require 'backports/2.0.0/array/bsearch'

This will enable Enumerable#flat_map and Array#bsearch, using the native versions if available or otherwise provide a pure Ruby version.

You can load many backports at once. For example, any version of Ruby up to today’s standards:

require 'backports'

This will bring in all the features of 1.8.7 and many features of Ruby 1.9.x and even Ruby 2.0.0 (for all versions of Ruby)!

Note: Although I am a Ruby committer, this gem is a personal project and is not endorsed by ruby-core.

What’s inside

Goals for backported features:

  1. Won’t break older code

  2. Pure Ruby (no C extensions)

  3. Pass RubySpec

Let’s be a bit more precise about the “breaking code” business. It is of course entirely possible that code will break, for example some core methods are monkeypatched or if the code relies on a certain call raising an exception. Example: [42].sample rescue "dum example" will return "dum example" without this gem and 42 with it.

A real incompatibility is, for example, Module::instance_methods which returns strings in 1.8 and symbols in 1.9. No change can be made without the risk of breaking existing code. Such incompatibilities are left unchanged, although you can require some of these changes in addition (see below)

All features of 1.8.7 are backported (well, almost all, see the exception list bellow), and many of the following versions up to 2.1

For historical reasons, some generic and self-contained features of active-support are also included. By simple I mean that String#camelcase is there, but #pluralize isn’t. These will probably be removed in the future, so it’s recommended to require those directly from active_support.

Installation & compatibility

backports is can be installed with:

(sudo) gem install backports

To use:

require 'rubygems'
# For only specific backports:
require 'backports/1.9.1/kernel/require_relative'
require 'backports/2.0.0/enumerable/lazy'

# For all backports up to a given version
require 'backports/1.9.2' # All backports for Ruby 1.9.2 and below

# Or for all backports
require 'backports'

Note: about a dozen of backports have a dependency that will be also loaded. For example, the backport of Enumerable#flat_map uses flatten(1), so if required from Ruby 1.8.6 (where Array#flatten does not accept an argument), the backport for Ruby’s 1.8.7 flatten with an argument will also be loaded.

With bundler, add to your Gemfile:

gem 'backports', :require => false

Run bundle install and require the desired backports.* Kernel

* +itself+

Compatible with Ru* Kernel

* +itself+y 1.8.6-2.4, JRuby and Rubinius.

Complete List of backports

Ruby 2.4 backports

  • Comparable

    • clamp

Ruby 2.3 backports

  • Array

    • bsearch_index

    • dig

  • Enumerable

    • chunk_while

    • grep_v

  • Hash

    • dig

    • fetch_values

    • to_proc

    • <=, <, >=, >

  • Numeric

    • negative?

    • positive?

  • String

    • unary + and -

  • Struct

    • dig

Ruby 2.2 backports

  • Enumerable

    • slice_after

    • slice_when

  • Float

    • prev_float

    • next_float

  • Kernel

    • itself

  • Method

    • curry

  • String

    • unicode_normalize

    • unicode_normalize!

    • unicode_normalize?

Ruby 2.1 backports

  • Array

    • to_h

  • Bignum

    • bit_length

  • Enumerable

    • to_h

  • Fixnum

    • bit_length

  • Module

    • include (now public)

Ruby 2.0 backports

  • Array

    • bsearch

  • Enumerable

    • lazy

  • Enumerator::Lazy

    • all methods

  • Hash

    • default_proc= (with nil argument)

    • to_h

  • nil.to_h

  • Range

    • bsearch

  • Struct

    • to_h

Ruby 1.9.3 backports

  • File

    • NULL

  • IO

    • advise (acts as a noop)

    • write, binwrite

  • String

    • byteslice

    • prepend

Ruby 1.9.2 backports

  • Array

    • rotate, rotate!

    • keep_if, select!

    • product (with block)

    • repeated_combination, repeated_permutation

    • sort_by!

    • uniq, uniq! (with block)

  • Complex

    • to_r

  • Dir

    • home

  • Enumerable

    • chunk

    • flat_map, collect_concat

    • join

    • slice_before

  • Float::INFINITY, NAN

  • Hash

    • keep_if, select!

  • Object

    • singleton_class

  • Random (new class)

Note: The methods of Random can’t be required individually; the class can only be required whole with require 'backports/1.9.2/random'.

Ruby 1.9.1 backports

Additionally, the following Ruby 1.9 features have been backported:

  • Array

    • try_convert

    • sample

  • Enumerable

    • each_with_object

    • each_with_index (with arguments)

  • Enumerator

    • new (with block)

  • File

    • binread

    • to_path

    • All class methods accepting filenames will accept files or anything with a #to_path method.

    • File.open accepts an options hash.

  • Float

    • round

  • Hash

    • assoc, rassoc

    • key

    • try_convert

    • default_proc=

  • Integer

    • magnitude

    • round

  • IO

    • bin_read

    • try_convert

    • ungetbyte

    • IO.open accepts an options hash.

  • Kernel

    • require_relative

  • Math

    • log (with base)

    • log2

  • Numeric

    • round

  • Object

    • define_singleton_method

    • public_method

    • public_send

  • Proc

    • yield

    • lambda?

    • curry

    • ===

  • Range

    • cover?

  • Regexp

    • try_convert

  • String

    • ascii_only?

    • chr

    • clear

    • codepoints, each_codepoint

    • get_byte, set_byte

    • ord

    • try_convert

Enumerator can be accessed directly (instead of Enumerable::Enumerator)

To include only these backports and those of the 1.8 line, require "backports/1.9.1".

Moreover, a pretty good imitation of BasicObject is available, but since it is only an imitation, it must be required explicitly:

require 'backports/basic_object'

Ruby 1.8.7

Complete Ruby 1.8.7 backporting (core language). Refer to the official list of changes. That’s about 130 backports!

Only exceptions:

  • String#gsub (the form returning an enumerator)

  • GC.stress= (not implemented)

  • Array#choice (removed in 1.9, use 1.9.1’s Array#sample instead)

Libraries

Libraries are slowly being backported. You simply require them as usual after requiring ‘backports/std_lib’. Requiring ‘backports/std_lib’ after the standard libraries is also supported.

require "backports/std_lib"
require "prime"
42.prime? # => false, even in Ruby 1.8.x

The following libraries are up to date with Ruby 1.9.3:

  • Matrix

  • Prime

  • Set

The following library is to date with Ruby 2.0.0:

  • OpenStruct (ostruct)

I am aware of the following backport gem, which probably won’t make it into this gem:

Requiring the whole of ‘backports’, or all backports for a given version of Ruby will also load ‘backports/std_lib’.

Forcing incompatibilities

Some backports would create incompatibilities in their current Ruby version but could be useful in some projects. It is possible to request such incompatible changes. Backports currently supports the following:

  • Hash

    • select (returns a Hash instead of an Array)

  • Enumerable / Array

    • map (returns an enumerator when called without a block)

  • String

    • length, size (for UTF-8 support)

These must be imported in addition to the backports gem, for example:

require "backports/force/hash_select"
{}.select{} # => {}, even in Ruby 1.8

Thanks

Thanks for the bug reports and patches, in particular the repeat offenders:

  • Arto Bendiken ( bendiken )

  • Konstantin Haase ( rkh)

  • Roger Pack ( rdp )

The best way to submit a patch is to also submit a patch to RubySpec and then a patch to backports that make it pass the spec. To test rubyspec:

git submodule init && git submodule update # => pulls rubyspecs
rake spec[array/bsearch]       # => tests Array#bsearch
rake spec[array/*]             # => tests all backported Array methods
rake spec  (or rake spec[*/*]) # => all rubyspecs for backported methods

Failures that are acceptable are added the to ‘tags` file.

License

backports is released under the terms of the MIT License, see the included LICENSE file.

Author

Marc-André Lafortune