Backports Library <img src=“https://travis-ci.org/marcandre/backports.png?branch=master”> <img src=“https://badge.fury.io/rb/backports.png” alt=“Gem Version” /> <img alt=‘Click here to lend your support to: Ruby Backports and make a donation at www.pledgie.com !’ src=‘http://www.pledgie.com/campaigns/19510.png?skin_name=chrome’ border=‘0’>¶ ↑
-
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.
Goals for backported features:
-
Won’t break older code
-
Pure Ruby (no C extensions)
-
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.
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.
-
Comparable
-
clamp
-
-
Array
-
bsearch_index
-
dig
-
-
Enumerable
-
chunk_while
-
grep_v
-
-
Hash
-
dig
-
fetch_values
-
to_proc
-
<=, <, >=, >
-
-
Numeric
-
negative?
-
positive?
-
-
String
-
unary + and -
-
-
Struct
-
dig
-
-
Enumerable
-
slice_after
-
slice_when
-
-
Float
-
prev_float
-
next_float
-
-
Kernel
-
itself
-
-
Method
-
curry
-
-
String
-
unicode_normalize
-
unicode_normalize!
-
unicode_normalize?
-
-
Array
-
to_h
-
-
Bignum
-
bit_length
-
-
Enumerable
-
to_h
-
-
Fixnum
-
bit_length
-
-
Module
-
include
(now public)
-
-
Array
-
bsearch
-
-
Enumerable
-
lazy
-
-
Enumerator::Lazy
-
all methods
-
-
Hash
-
default_proc=
(with nil argument) -
to_h
-
-
nil.to_h
-
Range
-
bsearch
-
-
Struct
-
to_h
-
-
File
-
NULL
-
-
IO
-
advise
(acts as a noop) -
write
,binwrite
-
-
String
-
byteslice
-
prepend
-
-
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'
.
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'
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 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:
-
Net::SMTP for Ruby 1.8.6: smtp_tls
Requiring the whole of ‘backports’, or all backports for a given version of Ruby will also load ‘backports/std_lib’.
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 for the bug reports and patches, in particular the repeat offenders:
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.
backports
is released under the terms of the MIT License, see the included LICENSE file.
- Author
-
Marc-André Lafortune