.. |
lib |
set_version |
spec |
test |
.document |
.gitignore |
.gitmodules |
.irbrc |
.travis.yml |
CHANGELOG.rdoc |
Gemfile |
Gemfile.lock |
LICENSE.txt |
README.rdoc |
Rakefile |
backports.gemspec |
default.mspec |
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.
Compatible with Ruby 1.8.6, 1.8.7, 1.9.1, 1.9.2, 1.9.3, 2.0, 2.1, JRuby and Rubinius.
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)
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, ideally (but not necessarily) after requiring backports.
require "backports" 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
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.
Marc-André Lafortune