plexus 0.5.4 → 0.5.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/Gemfile +3 -0
- data/LICENSE +37 -0
- data/README.md +208 -0
- data/Rakefile +25 -0
- data/lib/plexus.rb +90 -0
- data/lib/plexus/adjacency_graph.rb +225 -0
- data/lib/plexus/arc.rb +60 -0
- data/lib/plexus/arc_number.rb +50 -0
- data/lib/plexus/biconnected.rb +84 -0
- data/lib/plexus/chinese_postman.rb +91 -0
- data/lib/plexus/classes/graph_classes.rb +28 -0
- data/lib/plexus/common.rb +63 -0
- data/lib/plexus/comparability.rb +63 -0
- data/lib/plexus/directed_graph.rb +78 -0
- data/lib/plexus/directed_graph/algorithms.rb +95 -0
- data/lib/plexus/directed_graph/distance.rb +167 -0
- data/lib/plexus/dot.rb +94 -0
- data/lib/plexus/edge.rb +38 -0
- data/lib/plexus/ext.rb +79 -0
- data/lib/plexus/graph.rb +628 -0
- data/lib/plexus/graph_api.rb +35 -0
- data/lib/plexus/labels.rb +112 -0
- data/lib/plexus/maximum_flow.rb +77 -0
- data/lib/plexus/ruby_compatibility.rb +17 -0
- data/lib/plexus/search.rb +510 -0
- data/lib/plexus/strong_components.rb +93 -0
- data/lib/plexus/support/support.rb +9 -0
- data/lib/plexus/undirected_graph.rb +56 -0
- data/lib/plexus/undirected_graph/algorithms.rb +90 -0
- data/lib/plexus/version.rb +6 -0
- data/spec/biconnected_spec.rb +27 -0
- data/spec/chinese_postman_spec.rb +27 -0
- data/spec/community_spec.rb +44 -0
- data/spec/complement_spec.rb +27 -0
- data/spec/digraph_distance_spec.rb +121 -0
- data/spec/digraph_spec.rb +339 -0
- data/spec/dot_spec.rb +48 -0
- data/spec/edge_spec.rb +158 -0
- data/spec/inspection_spec.rb +38 -0
- data/spec/multi_edge_spec.rb +32 -0
- data/spec/neighborhood_spec.rb +36 -0
- data/spec/properties_spec.rb +146 -0
- data/spec/search_spec.rb +227 -0
- data/spec/spec.opts +4 -0
- data/spec/spec_helper.rb +59 -0
- data/spec/strong_components_spec.rb +61 -0
- data/spec/triangulated_spec.rb +125 -0
- data/spec/undirected_graph_spec.rb +220 -0
- data/vendor/priority-queue/CHANGELOG +33 -0
- data/vendor/priority-queue/Makefile +140 -0
- data/vendor/priority-queue/README +133 -0
- data/vendor/priority-queue/benchmark/dijkstra.rb +171 -0
- data/vendor/priority-queue/compare_comments.rb +49 -0
- data/vendor/priority-queue/doc/c-vs-rb.png +0 -0
- data/vendor/priority-queue/doc/compare_big.gp +14 -0
- data/vendor/priority-queue/doc/compare_big.png +0 -0
- data/vendor/priority-queue/doc/compare_small.gp +15 -0
- data/vendor/priority-queue/doc/compare_small.png +0 -0
- data/vendor/priority-queue/doc/results.csv +37 -0
- data/vendor/priority-queue/ext/priority_queue/CPriorityQueue/extconf.rb +2 -0
- data/vendor/priority-queue/ext/priority_queue/CPriorityQueue/priority_queue.c +947 -0
- data/vendor/priority-queue/lib/priority_queue.rb +14 -0
- data/vendor/priority-queue/lib/priority_queue/c_priority_queue.rb +1 -0
- data/vendor/priority-queue/lib/priority_queue/poor_priority_queue.rb +46 -0
- data/vendor/priority-queue/lib/priority_queue/ruby_priority_queue.rb +526 -0
- data/vendor/priority-queue/priority_queue.so +0 -0
- data/vendor/priority-queue/setup.rb +1551 -0
- data/vendor/priority-queue/test/priority_queue_test.rb +371 -0
- data/vendor/rdot.rb +360 -0
- metadata +100 -10
@@ -0,0 +1,95 @@
|
|
1
|
+
module Plexus
|
2
|
+
# Digraph is a directed graph which is a finite set of vertices
|
3
|
+
# and a finite set of edges connecting vertices. It cannot contain parallel
|
4
|
+
# edges going from the same source vertex to the same target. It also
|
5
|
+
# cannot contain loops, i.e. edges that go have the same vertex for source
|
6
|
+
# and target.
|
7
|
+
#
|
8
|
+
# DirectedPseudoGraph is a class that allows for parallel edges, and
|
9
|
+
# DirectedMultiGraph is a class that allows for parallel edges and loops
|
10
|
+
# as well.
|
11
|
+
module DirectedGraphBuilder
|
12
|
+
module Algorithms
|
13
|
+
include Search
|
14
|
+
include StrongComponents
|
15
|
+
include Distance
|
16
|
+
include ChinesePostman
|
17
|
+
|
18
|
+
# A directed graph is directed by definition.
|
19
|
+
#
|
20
|
+
# @return [Boolean] always true
|
21
|
+
#
|
22
|
+
def directed?
|
23
|
+
true
|
24
|
+
end
|
25
|
+
|
26
|
+
# A digraph uses the Arc class for edges.
|
27
|
+
#
|
28
|
+
# @return [Plexus::MultiArc, Plexus::Arc] `Plexus::MultiArc` if the graph allows for parallel edges,
|
29
|
+
# `Plexus::Arc` otherwise.
|
30
|
+
#
|
31
|
+
def edge_class
|
32
|
+
@parallel_edges ? Plexus::MultiArc : Plexus::Arc
|
33
|
+
end
|
34
|
+
|
35
|
+
# Reverse all edges in a graph.
|
36
|
+
#
|
37
|
+
# @return [DirectedGraph] a copy of the receiver for which the direction of edges has
|
38
|
+
# been inverted.
|
39
|
+
#
|
40
|
+
def reversal
|
41
|
+
result = self.class.new
|
42
|
+
edges.inject(result) { |a,e| a << e.reverse}
|
43
|
+
vertices.each { |v| result.add_vertex!(v) unless result.vertex?(v) }
|
44
|
+
result
|
45
|
+
end
|
46
|
+
|
47
|
+
# Check whether the graph is oriented or not.
|
48
|
+
#
|
49
|
+
# @return [Boolean]
|
50
|
+
#
|
51
|
+
def oriented?
|
52
|
+
e = edges
|
53
|
+
re = e.map { |x| x.reverse}
|
54
|
+
not e.any? { |x| re.include?(x)}
|
55
|
+
end
|
56
|
+
|
57
|
+
# Balanced is the state when the out edges count is equal to the in edges count.
|
58
|
+
#
|
59
|
+
# @return [Boolean]
|
60
|
+
#
|
61
|
+
def balanced?(v)
|
62
|
+
out_degree(v) == in_degree(v)
|
63
|
+
end
|
64
|
+
|
65
|
+
# Returns out_degree(v) - in_degree(v).
|
66
|
+
#
|
67
|
+
def delta(v)
|
68
|
+
out_degree(v) - in_degree(v)
|
69
|
+
end
|
70
|
+
|
71
|
+
def community(node, direction)
|
72
|
+
nodes, stack = {}, adjacent(node, :direction => direction)
|
73
|
+
while n = stack.pop
|
74
|
+
unless nodes[n.object_id] || node == n
|
75
|
+
nodes[n.object_id] = n
|
76
|
+
stack += adjacent(n, :direction => direction)
|
77
|
+
end
|
78
|
+
end
|
79
|
+
nodes.values
|
80
|
+
end
|
81
|
+
|
82
|
+
def descendants(node)
|
83
|
+
community(node, :out)
|
84
|
+
end
|
85
|
+
|
86
|
+
def ancestors(node)
|
87
|
+
community(node, :in)
|
88
|
+
end
|
89
|
+
|
90
|
+
def family(node)
|
91
|
+
community(node, :all)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
95
|
+
end
|
@@ -0,0 +1,167 @@
|
|
1
|
+
module Plexus
|
2
|
+
module DirectedGraphBuilder
|
3
|
+
|
4
|
+
# This module provides algorithms computing distance between
|
5
|
+
# vertices.
|
6
|
+
module Distance
|
7
|
+
|
8
|
+
# Shortest path computation.
|
9
|
+
#
|
10
|
+
# From: Jorgen Band-Jensen and Gregory Gutin,
|
11
|
+
# [*Digraphs: Theory, Algorithms and Applications*](http://www.springer.com/mathematics/numbers/book/978-1-84800-997-4), pg. 53-54.
|
12
|
+
# Complexity `O(n+m)`.
|
13
|
+
#
|
14
|
+
# Requires the graph to be acyclic. If the graph is not acyclic,
|
15
|
+
# then see {Distance#dijkstras_algorithm} or {Distance#bellman_ford_moore}
|
16
|
+
# for possible solutions.
|
17
|
+
#
|
18
|
+
# @param [vertex] start the starting vertex
|
19
|
+
# @param [Proc, #[]] weight can be a `Proc`, or anything else accessed using the `[]`
|
20
|
+
# operator. If not a `Proc`, and if no label accessible through `[]`, it will
|
21
|
+
# default to using the value stored in the label for the {Arc}. If a `Proc`, it will
|
22
|
+
# pass the edge to the proc and use the resulting value.
|
23
|
+
# @param [Integer] zero used for math systems with a different definition of zero
|
24
|
+
#
|
25
|
+
# @return [Hash] a hash with the key being a vertex and the value being the
|
26
|
+
# distance. A missing vertex from the hash is equivalent to an infinite distance.
|
27
|
+
def shortest_path(start, weight = nil, zero = 0)
|
28
|
+
dist = { start => zero }
|
29
|
+
path = {}
|
30
|
+
topsort(start) do |vi|
|
31
|
+
next if vi == start
|
32
|
+
dist[vi], path[vi] = adjacent(vi, :direction => :in).map do |vj|
|
33
|
+
[dist[vj] + cost(vj,vi,weight), vj]
|
34
|
+
end.min { |a,b| a[0] <=> b[0]}
|
35
|
+
end;
|
36
|
+
dist.keys.size == vertices.size ? [dist, path] : nil
|
37
|
+
end
|
38
|
+
|
39
|
+
# Finds the distance from a given vertex in a weighted digraph
|
40
|
+
# to the rest of the vertices, provided all the weights of arcs
|
41
|
+
# are non-negative.
|
42
|
+
#
|
43
|
+
# If negative arcs exist in the graph, two basic options exist:
|
44
|
+
#
|
45
|
+
# * modify all weights to be positive using an offset (temporary at least)
|
46
|
+
# * use the {Distance#bellman_ford_moore} algorithm.
|
47
|
+
#
|
48
|
+
# Also, if the graph is acyclic, use the {Distance#shortest_path algorithm}.
|
49
|
+
#
|
50
|
+
# From: Jorgen Band-Jensen and Gregory Gutin,
|
51
|
+
# [*Digraphs: Theory, Algorithms and Applications*](http://www.springer.com/mathematics/numbers/book/978-1-84800-997-4), pg. 53-54.
|
52
|
+
#
|
53
|
+
# Complexity `O(n*log(n) + m)`.
|
54
|
+
#
|
55
|
+
# @param [vertex] s
|
56
|
+
# @param [Proc, #[]] weight can be a `Proc`, or anything else accessed using the `[]`
|
57
|
+
# operator. If not a `Proc`, and if no label accessible through `[]`, it will
|
58
|
+
# default to using the value stored in the label for the {Arc}. If a `Proc`, it will
|
59
|
+
# pass the edge to the proc and use the resulting value.
|
60
|
+
# @param [Integer] zero used for math systems with a different definition of zero
|
61
|
+
# @return [Hash] a hash with the key being a vertex and the value being the
|
62
|
+
# distance. A missing vertex from the hash is equivalent to an infinite distance.
|
63
|
+
def dijkstras_algorithm(s, weight = nil, zero = 0)
|
64
|
+
q = vertices; distance = { s => zero }
|
65
|
+
path = {}
|
66
|
+
while not q.empty?
|
67
|
+
v = (q & distance.keys).inject(nil) { |a,k| (!a.nil?) && (distance[a] < distance[k]) ? a : k}
|
68
|
+
q.delete(v)
|
69
|
+
(q & adjacent(v)).each do |u|
|
70
|
+
c = cost(v, u, weight)
|
71
|
+
if distance[u].nil? or distance[u] > (c + distance[v])
|
72
|
+
distance[u] = c + distance[v]
|
73
|
+
path[u] = v
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
[distance, path]
|
78
|
+
end
|
79
|
+
|
80
|
+
# Finds the distances from a given vertex in a weighted digraph
|
81
|
+
# to the rest of the vertices, provided the graph has no negative cycle.
|
82
|
+
#
|
83
|
+
# If no negative weights exist, then {Distance#dijkstras_algorithm} is more
|
84
|
+
# efficient in time and space. Also, if the graph is acyclic, use the
|
85
|
+
# {Distance#shortest_path} algorithm.
|
86
|
+
#
|
87
|
+
# From: Jorgen Band-Jensen and Gregory Gutin,
|
88
|
+
# [*Digraphs: Theory, Algorithms and Applications*](http://www.springer.com/mathematics/numbers/book/978-1-84800-997-4), pg. 56-58..
|
89
|
+
#
|
90
|
+
# Complexity `O(nm)`.
|
91
|
+
#
|
92
|
+
# @param [vertex] s
|
93
|
+
# @param [Proc, #[]] weight can be a `Proc`, or anything else accessed using the `[]`
|
94
|
+
# operator. If not a `Proc`, and if no label accessible through `[]`, it will
|
95
|
+
# default to using the value stored in the label for the {Arc}. If a `Proc`, it will
|
96
|
+
# pass the edge to the proc and use the resulting value.
|
97
|
+
# @param [Integer] zero used for math systems with a different definition of zero
|
98
|
+
# @return [Hash] a hash with the key being a vertex and the value being the
|
99
|
+
# distance. A missing vertex from the hash is equivalent to an infinite distance.
|
100
|
+
def bellman_ford_moore(start, weight = nil, zero = 0)
|
101
|
+
distance = { start => zero }
|
102
|
+
path = {}
|
103
|
+
2.upto(vertices.size) do
|
104
|
+
edges.each do |e|
|
105
|
+
u, v = e[0], e[1]
|
106
|
+
unless distance[u].nil?
|
107
|
+
c = cost(u, v, weight) + distance[u]
|
108
|
+
if distance[v].nil? or c < distance[v]
|
109
|
+
distance[v] = c
|
110
|
+
path[v] = u
|
111
|
+
end
|
112
|
+
end
|
113
|
+
end
|
114
|
+
end
|
115
|
+
[distance, path]
|
116
|
+
end
|
117
|
+
|
118
|
+
# Uses the Floyd-Warshall algorithm to efficiently find
|
119
|
+
# and record shortest paths while establishing at the same time
|
120
|
+
# the costs for all vertices in a graph.
|
121
|
+
#
|
122
|
+
# See S.Skiena, *The Algorithm Design Manual*, Springer Verlag, 1998 for more details.
|
123
|
+
#
|
124
|
+
# O(n^3) complexity in time.
|
125
|
+
#
|
126
|
+
# @param [Proc, nil] weight specifies how an edge weight is determined.
|
127
|
+
# If it's a `Proc`, the {Arc} is passed to it; if it's `nil`, it will just use
|
128
|
+
# the value in the label for the Arc; otherwise the weight is
|
129
|
+
# determined by applying the `[]` operator to the value in the
|
130
|
+
# label for the {Arc}.
|
131
|
+
# @param [Integer] zero defines the zero value in the math system used.
|
132
|
+
# This allows for no assumptions to be made about the math system and
|
133
|
+
# fully functional duck typing.
|
134
|
+
# @return [Array(matrice, matrice, Hash)] a pair of matrices and a hash of delta values.
|
135
|
+
# The matrices will be indexed by two vertices and are implemented as a Hash of Hashes.
|
136
|
+
# The first matrix is the cost, the second matrix is the shortest path spanning tree.
|
137
|
+
# The delta (difference of number of in-edges and out-edges) is indexed by vertex.
|
138
|
+
def floyd_warshall(weight = nil, zero = 0)
|
139
|
+
c = Hash.new { |h,k| h[k] = Hash.new }
|
140
|
+
path = Hash.new { |h,k| h[k] = Hash.new }
|
141
|
+
delta = Hash.new { |h,k| h[k] = 0 }
|
142
|
+
edges.each do |e|
|
143
|
+
delta[e.source] += 1
|
144
|
+
delta[e.target] -= 1
|
145
|
+
path[e.source][e.target] = e.target
|
146
|
+
c[e.source][e.target] = cost(e, weight)
|
147
|
+
end
|
148
|
+
vertices.each do |k|
|
149
|
+
vertices.each do |i|
|
150
|
+
if c[i][k]
|
151
|
+
vertices.each do |j|
|
152
|
+
if c[k][j] &&
|
153
|
+
(c[i][j].nil? or c[i][j] > (c[i][k] + c[k][j]))
|
154
|
+
path[i][j] = path[i][k]
|
155
|
+
c[i][j] = c[i][k] + c[k][j]
|
156
|
+
return nil if i == j and c[i][j] < zero
|
157
|
+
end
|
158
|
+
end
|
159
|
+
end
|
160
|
+
end
|
161
|
+
end
|
162
|
+
[c, path, delta]
|
163
|
+
end
|
164
|
+
|
165
|
+
end # Distance
|
166
|
+
end # DirectedGraph
|
167
|
+
end # Plexus
|
data/lib/plexus/dot.rb
ADDED
@@ -0,0 +1,94 @@
|
|
1
|
+
module Plexus
|
2
|
+
module Dot
|
3
|
+
|
4
|
+
#FIXME: don't really understood where we stand with the dot generators.
|
5
|
+
# RDoc ships with a dot.rb which seems pretty efficient.
|
6
|
+
# Are these helpers still needed, and if not, how should we replace them?
|
7
|
+
|
8
|
+
# Creates a DOT::DOTDigraph for directed graphs or a DOT::DOTSubgraph for
|
9
|
+
# undirected graphs.
|
10
|
+
#
|
11
|
+
# @param [Hash] params can contain any graph property specified in
|
12
|
+
# rdot.rb. If an edge or vertex label is a kind of Hash, then the keys
|
13
|
+
# which match dot properties will be used as well.
|
14
|
+
# @return [DOT::DOTDigraph, DOT::DOTSubgraph]
|
15
|
+
def to_dot_graph(params = {})
|
16
|
+
params['name'] ||= self.class.name.gsub(/:/, '_')
|
17
|
+
fontsize = params['fontsize'] ? params['fontsize'] : '8'
|
18
|
+
graph = (directed? ? DOT::DOTDigraph : DOT::DOTSubgraph).new(params)
|
19
|
+
edge_klass = directed? ? DOT::DOTDirectedArc : DOT::DOTArc
|
20
|
+
|
21
|
+
vertices.each do |v|
|
22
|
+
name = v.to_s || v.__id__.to_s
|
23
|
+
name = name.dup.gsub(/"/, "'")
|
24
|
+
|
25
|
+
params = { 'name' => '"'+ name +'"',
|
26
|
+
'fontsize' => fontsize,
|
27
|
+
'label' => name}
|
28
|
+
|
29
|
+
v_label = vertex_label(v)
|
30
|
+
params.merge!(v_label) if v_label and v_label.kind_of? Hash
|
31
|
+
|
32
|
+
graph << DOT::DOTNode.new(params)
|
33
|
+
end
|
34
|
+
|
35
|
+
edges.each do |e|
|
36
|
+
if e.source.to_s.nil?
|
37
|
+
source_label = e.source.__id__.to_s
|
38
|
+
else
|
39
|
+
source_label = e.source.to_s.dup
|
40
|
+
end
|
41
|
+
|
42
|
+
if e.target.to_s.nil?
|
43
|
+
target_label = e.target.__id__.to_s
|
44
|
+
else
|
45
|
+
target_label = e.target.to_s.dup
|
46
|
+
end
|
47
|
+
|
48
|
+
source_label.gsub!(/"/, "'")
|
49
|
+
target_label.gsub!(/"/, "'")
|
50
|
+
|
51
|
+
params = { 'from' => '"'+ source_label + '"',
|
52
|
+
'to' => '"'+ target_label + '"',
|
53
|
+
'fontsize' => fontsize }
|
54
|
+
|
55
|
+
e_label = edge_label(e)
|
56
|
+
params.merge!(e_label) if e_label and e_label.kind_of? Hash
|
57
|
+
|
58
|
+
graph << edge_klass.new(params)
|
59
|
+
end
|
60
|
+
|
61
|
+
graph
|
62
|
+
end
|
63
|
+
|
64
|
+
# Output the dot format as a string
|
65
|
+
def to_dot(params = {})
|
66
|
+
to_dot_graph(params).to_s
|
67
|
+
end
|
68
|
+
|
69
|
+
# Call +dotty+ for the graph which is written to the file 'graph.dot'
|
70
|
+
# in the # current directory.
|
71
|
+
def dotty(params = {}, dotfile = 'graph.dot')
|
72
|
+
File.open(dotfile, 'w') {|f| f << to_dot(params) }
|
73
|
+
system('dotty', dotfile)
|
74
|
+
end
|
75
|
+
|
76
|
+
# Use +dot+ to create a graphical representation of the graph. Returns the
|
77
|
+
# filename of the graphics file.
|
78
|
+
def write_to_graphic_file(fmt = 'png', dotfile = 'graph')
|
79
|
+
src = dotfile + '.dot'
|
80
|
+
dot = dotfile + '.' + fmt
|
81
|
+
|
82
|
+
# DOT::DOTSubgraph creates subgraphs, but that's broken.
|
83
|
+
buffer = self.to_dot
|
84
|
+
buffer.gsub!(/^subgraph/, "graph")
|
85
|
+
|
86
|
+
File.open(src, 'w') {|f| f << buffer << "\n"}
|
87
|
+
system( "dot -T#{fmt} #{src} -o #{dot}" )
|
88
|
+
|
89
|
+
dot
|
90
|
+
end
|
91
|
+
alias as_dot_graphic write_to_graphic_file
|
92
|
+
|
93
|
+
end # Dot
|
94
|
+
end # module Plexus
|
data/lib/plexus/edge.rb
ADDED
@@ -0,0 +1,38 @@
|
|
1
|
+
module Plexus
|
2
|
+
# An undirected edge is simply an undirected pair (source, target) used in
|
3
|
+
# undirected graphs. Edge[u,v] == Edge[v,u]
|
4
|
+
class Edge < Arc
|
5
|
+
|
6
|
+
# Equality allows for the swapping of source and target
|
7
|
+
def eql?(other)
|
8
|
+
super or (self.class == other.class and target == other.source and source == other.target)
|
9
|
+
end
|
10
|
+
alias == eql?
|
11
|
+
|
12
|
+
# Hash is defined such that source and target can be reversed and the
|
13
|
+
# hash value will be the same
|
14
|
+
def hash
|
15
|
+
source.hash ^ target.hash
|
16
|
+
end
|
17
|
+
|
18
|
+
# Sort support
|
19
|
+
def <=>(rhs)
|
20
|
+
[[source,target].max, [source,target].min] <=> [[rhs.source,rhs.target].max, [rhs.source,rhs.target].min]
|
21
|
+
end
|
22
|
+
|
23
|
+
# Edge[1,2].to_s => "(1=2)"
|
24
|
+
# Edge[2,1].to_s => "(1=2)"
|
25
|
+
# Edge[2,1,'test'].to_s => "(1=2 test)"
|
26
|
+
def to_s
|
27
|
+
l = label ? " '#{label.to_s}'" : ''
|
28
|
+
s = source.to_s
|
29
|
+
t = target.to_s
|
30
|
+
"(#{[s,t].min}=#{[s,t].max}#{l})"
|
31
|
+
end
|
32
|
+
|
33
|
+
end
|
34
|
+
|
35
|
+
class MultiEdge < Edge
|
36
|
+
include ArcNumber
|
37
|
+
end
|
38
|
+
end
|
data/lib/plexus/ext.rb
ADDED
@@ -0,0 +1,79 @@
|
|
1
|
+
class Object
|
2
|
+
# Get the singleton class of the object.
|
3
|
+
# Depending on the object which requested its singleton class,
|
4
|
+
# a `module_eval` or a `class_eval` will be performed.
|
5
|
+
# Object of special constant type (`Fixnum`, `NilClass`, `TrueClass`,
|
6
|
+
# `FalseClass` and `Symbol`) return `nil` as they do not have a
|
7
|
+
# singleton class.
|
8
|
+
#
|
9
|
+
# @return the singleton class
|
10
|
+
def singleton_class
|
11
|
+
if self.respond_to? :module_eval
|
12
|
+
self.module_eval("class << self; self; end")
|
13
|
+
elsif self.respond_to? :instance_eval
|
14
|
+
begin
|
15
|
+
self.instance_eval("class << self; self; end")
|
16
|
+
rescue TypeError
|
17
|
+
nil
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
# Check wether the object is of the specified kind.
|
23
|
+
# If the receiver has a singleton class, will also perform
|
24
|
+
# the check on its singleton class' ancestors, so as to catch
|
25
|
+
# any included modules for object instances.
|
26
|
+
#
|
27
|
+
# Example:
|
28
|
+
#
|
29
|
+
# class A; include Digraph; end
|
30
|
+
# a.singleton_class.ancestors
|
31
|
+
# # => [Plexus::GraphAPI, Plexus::DirectedGraph::Algorithms, ...
|
32
|
+
# Plexus::Labels, Enumerable, Object, Plexus, Kernel, BasicObject]
|
33
|
+
# a.is_a? Plexus::Graph
|
34
|
+
# # => true
|
35
|
+
#
|
36
|
+
# @param [Class] klass
|
37
|
+
# @return [Boolean]
|
38
|
+
def is_a? klass
|
39
|
+
sc = self.singleton_class
|
40
|
+
if not sc.nil?
|
41
|
+
self.singleton_class.ancestors.include?(klass) || super
|
42
|
+
else
|
43
|
+
super
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
class Module
|
49
|
+
# Helper which purpose is, given a class including a module,
|
50
|
+
# to make each methods defined within a module's submodule `ClassMethods`
|
51
|
+
# available as class methods to the receiving class.
|
52
|
+
#
|
53
|
+
# Example:
|
54
|
+
#
|
55
|
+
# module A
|
56
|
+
# extends_host
|
57
|
+
# module ClassMethods
|
58
|
+
# def selfy; puts "class method for #{self}"; end
|
59
|
+
# end
|
60
|
+
# end
|
61
|
+
#
|
62
|
+
# class B; include A; end
|
63
|
+
#
|
64
|
+
# B.selfy
|
65
|
+
# # => class method for B
|
66
|
+
#
|
67
|
+
# @option *params [Symbol] :with (:ClassMethods) the name of the
|
68
|
+
# module to extend the receiver with
|
69
|
+
def extends_host(*params)
|
70
|
+
args = (params.pop if params.last.is_a? Hash) || {}
|
71
|
+
@_extension_module = args[:with] || :ClassMethods
|
72
|
+
|
73
|
+
def included(base)
|
74
|
+
unless @_extension_module.nil?
|
75
|
+
base.extend(self.const_get(@_extension_module))
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|