Build and manipulate tree structures in Ruby.
Synopsis
Jumoku provides you with tree structures and related tools to perform manipulation and computation, the easy way. Trees are frequently used to mimic hierarchal structures such as filesystems, or modelize decisionnal patterns, for instance.
Jumoku is built upon Plexus, a ruby-powered Graph Theory library, and aims at being a fully-fledged, pithy solution for tree-like structures managment. See below for additionnal information about graphs, trees, arborescences, why they're different and how to make good use of them.
A few words about trees
A Tree is a graph subject to three basic constraints: nodes are all connected, they must not form any loop, and the branches binding nodes have no preferential direction. Trees are an important subset of graphs used in a whole slew of computer science and mathematical problems: network modelization, datasets storage, scientific computation, load balancing, games, AI designs, etc. They however are not limited to directed patterns: a tree is not compelled to have a "root" node, nor to have "leaves" as you may think prima facie. Trees with such features are called arborescences and Jumoku has support for them, too.
Jumoku (currently under early development stage) provides you with the following structures:
- RawUndirectedTree: a pure tree-graph with only basic features
- RawDirectedTree: a tree that's an arborescence, that is, a tree with a flow from its root to its leaf nodes
- Tree: a tree-graph with extended features built upon RawUndirectedTree. That's what you'd want to use as a fully-fledged tree structure
- Arborescence: an arbo-graph with extended features built upon RawDirectedTree. That's what you'd want to use as a fully-fledged arborescence structure
- AVLTree (not yet)
- RedBlackTree (not yet)
You can also extend those structures with hybrid behaviors (not Graph Theory compliant but may be useful):
- Directed (not yet): relax the undirected constraint
- Loopy (not yet): relax the acyclic constraint
- Atomic (not yet): relax the connected constraint
There are also strategies one may enable:
- a simple edge labeling scheme (increasing integer indexes), providing edges and nodes sorting facilities
Basic usage
To create an instance of a tree, you may use either inheritance or mixins.
Inheritance or direct initialization
class MyTree < Jumoku::Tree; end
tree = MyTree.new
# or even simpler:
tree = Jumoku::Tree.new
Mixing-in a "builder" module
class MyTree
include Jumoku::TreeBuilder
end
tree = MyTree.new
The RawTree class is actually implemented this way.
What you get
Following the previous code example, tree
is now a Tree object, shipping with some default options:
- it is a valid tree per se (an undirected, acyclic, connected graph)
- Jumoku API's methods will ensure it remains so as you manipulate it
- it has no constraint on the number of branches per node (its branching number)
Install, first steps and bootstraping
In order to play with Jumoku, you must:
gem install jumoku
Now you can play in IRB:
$ irb
ruby-1.9.1-p378 > require 'jumoku'
=> true
ruby-1.9.1-p378 > include Jumoku # so you won't have to prefix everything with "Jumoku::"
=> Object
ruby-1.9.1-p378 > t = Tree.new
=> #<Jumoku::Tree:0x000000020d5ac8 @vertex_dict={}, @vertex_labels={}, @edge_labels={}, @allow_loops=false, @parallel_edges=false, @edgelist_class=Set>
ruby-1.9.1-p378 > t.methods
=> # lot of stuff hopefully :)
A good way to get you started is by reading the doc online. You can locally generate the doc with any of the following commands (you'll need to have the yard
gem installed):
rake yard
yardoc
Be sure to take a look at the tests in spec/
as well, as they document the public API extensively.
What about graphs?
Trees are just simple graphs, with specific constraints on edges (branches) and vertices (nodes). In graph theory, a tree is defined as a graph which is undirected, acyclic and connected. A forest is a disjoint union of trees. Let's review those constraints:
- undirected: the branches of a tree have no direction and you can flow either from root toward leaves or reversily;
- acyclic: a cycle is defined as a path such that the starting node and the ending node are the same. Such paths are forbidden in a tree;
- connected: a tree is such that pair of distinct nodes can be connected through some path (but not a cycle).
This is quite restrictive, but not that restrictive. Instinctively, a tree structure would rather be described as an arborescence, that is a collection of nodes with a root node and some leaves (ending nodes), that is with a general direction (top-down). That is what you would use to use to modelize nested directories, for instance. From the mathematical point of view, legacy trees and arborescences have some little differencies (the latter are more constrained). Jumoku provides both structures.
Several ruby graph libraries exist, some pure-ruby, some as bindings (for instance, the popular igraph C backend). Jumoku makes use of the most up-to-date project among those available, Plexus. It's been forked to implement new features required by Jumoku, and is vendorized within this gem.
License
MIT License. See the LICENSE file.
Contributing
Fork, hack, request!