Gem Version Build Status Code Climate Coverage Status MIT licensed

nio4r: New IO for Ruby.

nio4r provides an abstract, cross-platform stateful I/O selector API for Ruby. I/O selectors are the heart of "reactor"-based event loops, and monitor multiple I/O objects for various types of readiness, e.g. ready for reading or writing.

The most similar API provided by Ruby today is, however the select API requires you to pass in arrays of all of the I/O objects you're interested in every time. nio4r provides a more object-oriented API that lets you register I/O objects with a selector then handle them when they're selected for various types of events.

nio4r is modeled after the Java NIO API, but simplified for ease-of-use.

Its goals are:

  • Expose high-level interfaces for stateful IO selectors
  • Keep the API small to maximize both portability and performance across many different OSes and Ruby VMs
  • Provide inherently thread-safe facilities for working with IO objects

Celluloid::IO uses nio4r to monitor multiple IO objects from a single Celluloid actor.

Supported Platforms

nio4r is known to work on the following Ruby implementations:

  • MRI/YARV 1.9.3, 2.0.0, 2.1.0, 2.2.0, 2.3.0
  • JRuby 1.7.x
  • Rubinius 2.x
  • A pure Ruby implementation based on is also provided

Platform notes:

  • MRI/YARV and Rubinius implement nio4r with a C extension based on libev, which provides a high performance binding to native IO APIs
  • JRuby uses a Java extension based on the high performance Java NIO subsystem
  • A pure Ruby implementation is also provided for Ruby implementations which don't implement the MRI C extension API



The NIO::Selector class is the main API provided by nio4r. Use it where you might otherwise use, but want to monitor the same set of IO objects across multiple select calls rather than having to reregister them every single time:

require 'nio'

selector =

To monitor IO objects, attach them to the selector with the NIO::Selector#register method, monitoring them for read readiness with the :r parameter, write readiness with the :w parameter, or both with :rw.

>> reader, writer = IO.pipe
 => [#<IO:0xf30>, #<IO:0xf34>]
>> monitor = selector.register(reader, :r)
 => #<NIO::Monitor:0xfbc>

After registering an IO object with the selector, you'll get a NIO::Monitor object which you can use for managing how a particular IO object is being monitored. Monitors will store an arbitrary value of your choice, which provides an easy way to implement callbacks:

>> monitor = selector.register(reader, :r)
 => #<NIO::Monitor:0xfbc>
>> monitor.value = proc { puts "Got some data: #{}" }
 => #<Proc:0x1000@(irb):4>

The main method of importance is NIO::Selector#select, which monitors all registered IO objects and returns an array of monitors that are ready.

>> writer << "Hi there!"
 => #<IO:0x103c>
>> ready =
 => [#<NIO::Monitor:0xfbc>]
>> ready.each { |m| }
Got some data: Hi there!
 => [#<NIO::Monitor:0xfbc>]

By default, NIO::Selector#select will block indefinitely until one of the IO objects being monitored becomes ready. However, you can also pass a timeout to wait in seconds to NIO::Selector#select just like you can with

ready = # Wait 15 seconds

If a timeout occurs, ready will be nil.

You can avoid allocating an array each time you call NIO::Selector#select by passing a block to select. The block will be called for each ready monitor object, with that object passed as an argument. The number of ready monitors is returned as a Fixnum:

>> { |m| }
Got some data: Hi there!
 => 1

When you're done monitoring a particular IO object, just deregister it from the selector:



Monitors provide methods which let you introspect on why a particular IO object was selected. These methods are not thread safe unless you are holding the selector lock (i.e. if you're in a block passed to #select). Only use them if you aren't concerned with thread safety, or you're within a #select block:

  • #interests: what this monitor is interested in (:r, :w, or :rw)
  • #interests=: change the current interests for a monitor (to :r, :w, or :rw)
  • #readiness: what I/O operations the monitored object is ready for
  • #readable?: was the IO readable last time it was selected?
  • #writable?: was the IO writable last time it was selected?

Monitors also support a #value and #value= method for storing a handle to an arbitrary object of your choice (e.g. a proc)

Flow Control

For information on how to compose nio4r selectors inside of event loops, please read the Flow Control Guide on the Wiki


nio4r provides internal locking to ensure that it's safe to use from multiple concurrent threads. Only one thread can select on a NIO::Selector at a given time, and while a thread is selecting other threads are blocked from registering or deregistering IO objects. Once a pending select has completed, requests to register/unregister IO objects will be processed.

NIO::Selector#wakeup allows one thread to unblock another thread that's in the middle of an NIO::Selector#select operation. This lets other threads that need to communicate immediately with the selector unblock it so it can process other events that it's not presently selecting on.

What nio4r is not

nio4r is not a full-featured event framework like EventMachine or Instead, nio4r is the sort of thing you might write a library like that on top of. nio4r provides a minimal API such that individual Ruby implementers may choose to produce optimized versions for their platform, without having to maintain a large codebase.

As of the time of writing, the current implementation is (approximately):

  • 200 lines of Ruby code
  • 700 lines of "custom" C code (not counting libev)
  • 400 lines of Java code

nio4r is also not a replacement for Kinder Gentler IO (KGIO), a set of advanced Ruby IO APIs. At some point in the future nio4r might provide a cross-platform implementation that uses KGIO on CRubies, and Java NIO on JRuby, however this is not the case today.


Copyright (c) 2011-2016 Tony Arcieri. Distributed under the MIT License. See LICENSE.txt for further details.

Includes libev 4.22. Copyright (c) 2007-2015 Marc Alexander Lehmann. Distributed under the BSD license. See ext/libev/LICENSE for details.