factorylabs-railroad 0.6.0.1
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/COPYING +340 -0
- data/ChangeLog +85 -0
- data/History.txt +6 -0
- data/Manifest.txt +17 -0
- data/README.txt +145 -0
- data/Rakefile +15 -0
- data/bin/railroad +45 -0
- data/init.rb +0 -0
- data/lib/railroad.rb +11 -0
- data/lib/railroad/aasm_diagram.rb +79 -0
- data/lib/railroad/app_diagram.rb +86 -0
- data/lib/railroad/controllers_diagram.rb +81 -0
- data/lib/railroad/diagram_graph.rb +133 -0
- data/lib/railroad/models_diagram.rb +171 -0
- data/lib/railroad/options_struct.rb +169 -0
- data/tasks/diagrams.rake +18 -0
- metadata +79 -0
@@ -0,0 +1,133 @@
|
|
1
|
+
# RailRoad - RoR diagrams generator
|
2
|
+
# http://railroad.rubyforge.org
|
3
|
+
#
|
4
|
+
# Copyright 2007-2008 - Javier Smaldone (http://www.smaldone.com.ar)
|
5
|
+
# See COPYING for more details
|
6
|
+
|
7
|
+
|
8
|
+
# RailRoad diagram structure
|
9
|
+
class DiagramGraph
|
10
|
+
|
11
|
+
def initialize
|
12
|
+
@diagram_type = ''
|
13
|
+
@show_label = false
|
14
|
+
@nodes = []
|
15
|
+
@edges = []
|
16
|
+
end
|
17
|
+
|
18
|
+
def add_node(node)
|
19
|
+
@nodes << node
|
20
|
+
end
|
21
|
+
|
22
|
+
def add_edge(edge)
|
23
|
+
@edges << edge
|
24
|
+
end
|
25
|
+
|
26
|
+
def diagram_type= (type)
|
27
|
+
@diagram_type = type
|
28
|
+
end
|
29
|
+
|
30
|
+
def show_label= (value)
|
31
|
+
@show_label = value
|
32
|
+
end
|
33
|
+
|
34
|
+
|
35
|
+
# Generate DOT graph
|
36
|
+
def to_dot
|
37
|
+
return dot_header +
|
38
|
+
@nodes.map{|n| dot_node n[0], n[1], n[2]}.join +
|
39
|
+
@edges.map{|e| dot_edge e[0], e[1], e[2], e[3]}.join +
|
40
|
+
dot_footer
|
41
|
+
end
|
42
|
+
|
43
|
+
# Generate XMI diagram (not yet implemented)
|
44
|
+
def to_xmi
|
45
|
+
STDERR.print "Sorry. XMI output not yet implemented.\n\n"
|
46
|
+
return ""
|
47
|
+
end
|
48
|
+
|
49
|
+
private
|
50
|
+
|
51
|
+
# Build DOT diagram header
|
52
|
+
def dot_header
|
53
|
+
result = "digraph #{@diagram_type.downcase}_diagram {\n" +
|
54
|
+
"\tgraph[overlap=false, splines=true]\n"
|
55
|
+
result += dot_label if @show_label
|
56
|
+
return result
|
57
|
+
end
|
58
|
+
|
59
|
+
# Build DOT diagram footer
|
60
|
+
def dot_footer
|
61
|
+
return "}\n"
|
62
|
+
end
|
63
|
+
|
64
|
+
# Build diagram label
|
65
|
+
def dot_label
|
66
|
+
return "\t_diagram_info [shape=\"plaintext\", " +
|
67
|
+
"label=\"#{@diagram_type} diagram\\l" +
|
68
|
+
"Date: #{Time.now.strftime "%b %d %Y - %H:%M"}\\l" +
|
69
|
+
"Migration version: " +
|
70
|
+
"#{ActiveRecord::Migrator.current_version}\\l" +
|
71
|
+
"Generated by #{APP_HUMAN_NAME} #{APP_VERSION.join('.')}"+
|
72
|
+
"\\l\", fontsize=14]\n"
|
73
|
+
end
|
74
|
+
|
75
|
+
# Build a DOT graph node
|
76
|
+
def dot_node(type, name, attributes=nil)
|
77
|
+
case type
|
78
|
+
when 'model'
|
79
|
+
options = 'shape=Mrecord, label="{' + name + '|'
|
80
|
+
options += attributes.join('\l')
|
81
|
+
options += '\l}"'
|
82
|
+
when 'model-brief'
|
83
|
+
options = ''
|
84
|
+
when 'class'
|
85
|
+
options = 'shape=record, label="{' + name + '|}"'
|
86
|
+
when 'class-brief'
|
87
|
+
options = 'shape=box'
|
88
|
+
when 'controller'
|
89
|
+
options = 'shape=Mrecord, label="{' + name + '|'
|
90
|
+
public_methods = attributes[:public].join('\l')
|
91
|
+
protected_methods = attributes[:protected].join('\l')
|
92
|
+
private_methods = attributes[:private].join('\l')
|
93
|
+
options += public_methods + '\l|' + protected_methods + '\l|' +
|
94
|
+
private_methods + '\l'
|
95
|
+
options += '}"'
|
96
|
+
when 'controller-brief'
|
97
|
+
options = ''
|
98
|
+
when 'module'
|
99
|
+
options = 'shape=box, style=dotted, label="' + name + '"'
|
100
|
+
when 'aasm'
|
101
|
+
# Return subgraph format
|
102
|
+
return "subgraph cluster_#{name.downcase} {\n\tlabel = #{quote(name)}\n\t#{attributes.join("\n ")}}"
|
103
|
+
end # case
|
104
|
+
return "\t#{quote(name)} [#{options}]\n"
|
105
|
+
end # dot_node
|
106
|
+
|
107
|
+
# Build a DOT graph edge
|
108
|
+
def dot_edge(type, from, to, name = '')
|
109
|
+
options = name != '' ? "label=\"#{name}\", " : ''
|
110
|
+
case type
|
111
|
+
when 'one-one'
|
112
|
+
#options += 'taillabel="1"'
|
113
|
+
options += 'arrowtail=odot, arrowhead=dot, dir=both'
|
114
|
+
when 'one-many'
|
115
|
+
#options += 'taillabel="n"'
|
116
|
+
options += 'arrowtail=crow, arrowhead=dot, dir=both'
|
117
|
+
when 'many-many'
|
118
|
+
#options += 'taillabel="n", headlabel="n", arrowtail="normal"'
|
119
|
+
options += 'arrowtail=crow, arrowhead=crow, dir=both'
|
120
|
+
when 'is-a'
|
121
|
+
options += 'arrowhead="none", arrowtail="onormal"'
|
122
|
+
when 'event'
|
123
|
+
options += "fontsize=10"
|
124
|
+
end
|
125
|
+
return "\t#{quote(from)} -> #{quote(to)} [#{options}]\n"
|
126
|
+
end # dot_edge
|
127
|
+
|
128
|
+
# Quotes a class name
|
129
|
+
def quote(name)
|
130
|
+
'"' + name.to_s + '"'
|
131
|
+
end
|
132
|
+
|
133
|
+
end # class DiagramGraph
|
@@ -0,0 +1,171 @@
|
|
1
|
+
# RailRoad - RoR diagrams generator
|
2
|
+
# http://railroad.rubyforge.org
|
3
|
+
#
|
4
|
+
# Copyright 2007-2008 - Javier Smaldone (http://www.smaldone.com.ar)
|
5
|
+
# See COPYING for more details
|
6
|
+
|
7
|
+
# RailRoad models diagram
|
8
|
+
class ModelsDiagram < AppDiagram
|
9
|
+
|
10
|
+
def initialize(options)
|
11
|
+
#options.exclude.map! {|e| "app/models/" + e}
|
12
|
+
super options
|
13
|
+
@graph.diagram_type = 'Models'
|
14
|
+
# Processed habtm associations
|
15
|
+
@habtm = []
|
16
|
+
end
|
17
|
+
|
18
|
+
# Process model files
|
19
|
+
def generate
|
20
|
+
STDERR.print "Generating models diagram\n" if @options.verbose
|
21
|
+
base = "app/models/"
|
22
|
+
files = Dir.glob("app/models/**/*.rb")
|
23
|
+
files += Dir.glob("vendor/plugins/**/app/models/*.rb") if @options.plugins_models
|
24
|
+
files -= @options.exclude
|
25
|
+
files.each do |file|
|
26
|
+
model_name = file.gsub(/^#{base}([\w_\/\\]+)\.rb/, '\1')
|
27
|
+
# Hack to skip all xxx_related.rb files
|
28
|
+
next if /_related/i =~ model_name
|
29
|
+
|
30
|
+
klass = begin
|
31
|
+
model_name.classify.constantize
|
32
|
+
rescue LoadError
|
33
|
+
model_name.gsub!(/.*[\/\\]/, '')
|
34
|
+
retry
|
35
|
+
rescue NameError
|
36
|
+
next
|
37
|
+
end
|
38
|
+
|
39
|
+
process_class klass
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
private
|
44
|
+
|
45
|
+
# Load model classes
|
46
|
+
def load_classes
|
47
|
+
begin
|
48
|
+
disable_stdout
|
49
|
+
files = Dir.glob("app/models/**/*.rb")
|
50
|
+
files += Dir.glob("vendor/plugins/**/app/models/*.rb") if @options.plugins_models
|
51
|
+
files -= @options.exclude
|
52
|
+
files.each do |m|
|
53
|
+
require m
|
54
|
+
end
|
55
|
+
enable_stdout
|
56
|
+
rescue LoadError
|
57
|
+
enable_stdout
|
58
|
+
print_error "model classes"
|
59
|
+
raise
|
60
|
+
end
|
61
|
+
end # load_classes
|
62
|
+
|
63
|
+
# Process a model class
|
64
|
+
def process_class(current_class)
|
65
|
+
|
66
|
+
STDERR.print "\tProcessing #{current_class}\n" if @options.verbose
|
67
|
+
|
68
|
+
generated = false
|
69
|
+
|
70
|
+
# Is current_clas derived from ActiveRecord::Base?
|
71
|
+
if current_class.respond_to?'reflect_on_all_associations'
|
72
|
+
|
73
|
+
|
74
|
+
node_attribs = []
|
75
|
+
if @options.brief || current_class.abstract_class? || current_class.superclass != ActiveRecord::Base
|
76
|
+
node_type = 'model-brief'
|
77
|
+
else
|
78
|
+
node_type = 'model'
|
79
|
+
|
80
|
+
# Collect model's content columns
|
81
|
+
|
82
|
+
content_columns = current_class.content_columns
|
83
|
+
|
84
|
+
if @options.hide_magic
|
85
|
+
magic_fields = [
|
86
|
+
# Restful Authentication
|
87
|
+
"login", "crypted_password", "salt", "remember_token", "remember_token_expires_at", "activation_code", "activated_at",
|
88
|
+
# From patch #13351
|
89
|
+
# http://wiki.rubyonrails.org/rails/pages/MagicFieldNames
|
90
|
+
"created_at", "created_on", "updated_at", "updated_on",
|
91
|
+
"lock_version", "type", "id", "position", "parent_id", "lft",
|
92
|
+
"rgt", "quote", "template"
|
93
|
+
]
|
94
|
+
magic_fields << current_class.table_name + "_count" if current_class.respond_to? 'table_name'
|
95
|
+
content_columns = current_class.content_columns.select {|c| ! magic_fields.include? c.name}
|
96
|
+
else
|
97
|
+
content_columns = current_class.content_columns
|
98
|
+
end
|
99
|
+
|
100
|
+
content_columns.each do |a|
|
101
|
+
content_column = a.name
|
102
|
+
content_column += ' :' + a.type.to_s unless @options.hide_types
|
103
|
+
node_attribs << content_column
|
104
|
+
end
|
105
|
+
end
|
106
|
+
@graph.add_node [node_type, current_class.name, node_attribs]
|
107
|
+
generated = true
|
108
|
+
# Process class associations
|
109
|
+
associations = current_class.reflect_on_all_associations
|
110
|
+
if @options.inheritance && ! @options.transitive
|
111
|
+
superclass_associations = current_class.superclass.reflect_on_all_associations
|
112
|
+
|
113
|
+
associations = associations.select{|a| ! superclass_associations.include? a}
|
114
|
+
# This doesn't works!
|
115
|
+
# associations -= current_class.superclass.reflect_on_all_associations
|
116
|
+
end
|
117
|
+
associations.each do |a|
|
118
|
+
process_association current_class.name, a
|
119
|
+
end
|
120
|
+
elsif @options.all && (current_class.is_a? Class)
|
121
|
+
# Not ActiveRecord::Base model
|
122
|
+
node_type = @options.brief ? 'class-brief' : 'class'
|
123
|
+
@graph.add_node [node_type, current_class.name]
|
124
|
+
generated = true
|
125
|
+
elsif @options.modules && (current_class.is_a? Module)
|
126
|
+
@graph.add_node ['module', current_class.name]
|
127
|
+
end
|
128
|
+
|
129
|
+
# Only consider meaningful inheritance relations for generated classes
|
130
|
+
if @options.inheritance && generated &&
|
131
|
+
(current_class.superclass != ActiveRecord::Base) &&
|
132
|
+
(current_class.superclass != Object)
|
133
|
+
@graph.add_edge ['is-a', current_class.superclass.name, current_class.name]
|
134
|
+
end
|
135
|
+
|
136
|
+
end # process_class
|
137
|
+
|
138
|
+
# Process a model association
|
139
|
+
def process_association(class_name, assoc)
|
140
|
+
|
141
|
+
STDERR.print "\t\tProcessing model association #{assoc.name.to_s}\n" if @options.verbose
|
142
|
+
|
143
|
+
# Skip "belongs_to" associations
|
144
|
+
return if assoc.macro.to_s == 'belongs_to'
|
145
|
+
|
146
|
+
# Only non standard association names needs a label
|
147
|
+
|
148
|
+
# from patch #12384
|
149
|
+
# if assoc.class_name == assoc.name.to_s.singularize.camelize
|
150
|
+
assoc_class_name = (assoc.class_name.respond_to? 'underscore') ? assoc.class_name.underscore.singularize.camelize : assoc.class_name
|
151
|
+
if assoc_class_name == assoc.name.to_s.singularize.camelize
|
152
|
+
assoc_name = ''
|
153
|
+
else
|
154
|
+
assoc_name = assoc.name.to_s
|
155
|
+
end
|
156
|
+
# STDERR.print "#{assoc_name}\n"
|
157
|
+
if assoc.macro.to_s == 'has_one'
|
158
|
+
assoc_type = 'one-one'
|
159
|
+
elsif assoc.macro.to_s == 'has_many' && (! assoc.options[:through])
|
160
|
+
assoc_type = 'one-many'
|
161
|
+
else # habtm or has_many, :through
|
162
|
+
return if @habtm.include? [assoc.class_name, class_name, assoc_name]
|
163
|
+
assoc_type = 'many-many'
|
164
|
+
@habtm << [class_name, assoc.class_name, assoc_name]
|
165
|
+
end
|
166
|
+
# from patch #12384
|
167
|
+
# @graph.add_edge [assoc_type, class_name, assoc.class_name, assoc_name]
|
168
|
+
@graph.add_edge [assoc_type, class_name, assoc_class_name, assoc_name]
|
169
|
+
end # process_association
|
170
|
+
|
171
|
+
end # class ModelsDiagram
|
@@ -0,0 +1,169 @@
|
|
1
|
+
# RailRoad - RoR diagrams generator
|
2
|
+
# http://railroad.rubyforge.org
|
3
|
+
#
|
4
|
+
# Copyright 2007-2008 - Javier Smaldone (http://www.smaldone.com.ar)
|
5
|
+
# See COPYING for more details
|
6
|
+
|
7
|
+
require 'ostruct'
|
8
|
+
|
9
|
+
# RailRoad command line options parser
|
10
|
+
class OptionsStruct < OpenStruct
|
11
|
+
|
12
|
+
require 'optparse'
|
13
|
+
|
14
|
+
def initialize
|
15
|
+
init_options = { :all => false,
|
16
|
+
:brief => false,
|
17
|
+
:exclude => [],
|
18
|
+
:inheritance => false,
|
19
|
+
:join => false,
|
20
|
+
:label => false,
|
21
|
+
:modules => false,
|
22
|
+
:hide_magic => false,
|
23
|
+
:hide_types => false,
|
24
|
+
:hide_public => false,
|
25
|
+
:hide_protected => false,
|
26
|
+
:hide_private => false,
|
27
|
+
:plugins_models => false,
|
28
|
+
:root => '',
|
29
|
+
:transitive => false,
|
30
|
+
:verbose => false,
|
31
|
+
:xmi => false,
|
32
|
+
:command => '' }
|
33
|
+
super(init_options)
|
34
|
+
end # initialize
|
35
|
+
|
36
|
+
def parse(args)
|
37
|
+
@opt_parser = OptionParser.new do |opts|
|
38
|
+
opts.banner = "Usage: #{APP_NAME} [options] command"
|
39
|
+
opts.separator ""
|
40
|
+
opts.separator "Common options:"
|
41
|
+
opts.on("-b", "--brief", "Generate compact diagram",
|
42
|
+
" (no attributes nor methods)") do |b|
|
43
|
+
self.brief = b
|
44
|
+
end
|
45
|
+
opts.on("-e", "--exclude file1[,fileN]", Array, "Exclude given files") do |list|
|
46
|
+
self.exclude = list
|
47
|
+
end
|
48
|
+
opts.on("-i", "--inheritance", "Include inheritance relations") do |i|
|
49
|
+
self.inheritance = i
|
50
|
+
end
|
51
|
+
opts.on("-l", "--label", "Add a label with diagram information",
|
52
|
+
" (type, date, migration, version)") do |l|
|
53
|
+
self.label = l
|
54
|
+
end
|
55
|
+
opts.on("-o", "--output FILE", "Write diagram to file FILE") do |f|
|
56
|
+
self.output = f
|
57
|
+
end
|
58
|
+
opts.on("-r", "--root PATH", "Set PATH as the application root") do |r|
|
59
|
+
self.root = r
|
60
|
+
end
|
61
|
+
opts.on("-v", "--verbose", "Enable verbose output",
|
62
|
+
" (produce messages to STDOUT)") do |v|
|
63
|
+
self.verbose = v
|
64
|
+
end
|
65
|
+
opts.on("-x", "--xmi", "Produce XMI instead of DOT",
|
66
|
+
" (for UML tools)") do |x|
|
67
|
+
self.xmi = x
|
68
|
+
end
|
69
|
+
opts.separator ""
|
70
|
+
opts.separator "Models diagram options:"
|
71
|
+
opts.on("-a", "--all", "Include all models",
|
72
|
+
" (not only ActiveRecord::Base derived)") do |a|
|
73
|
+
self.all = a
|
74
|
+
end
|
75
|
+
opts.on("--hide-magic", "Hide magic field names") do |h|
|
76
|
+
self.hide_magic = h
|
77
|
+
end
|
78
|
+
opts.on("--hide-types", "Hide attributes type") do |h|
|
79
|
+
self.hide_types = h
|
80
|
+
end
|
81
|
+
opts.on("-j", "--join", "Concentrate edges") do |j|
|
82
|
+
self.join = j
|
83
|
+
end
|
84
|
+
opts.on("-m", "--modules", "Include modules") do |m|
|
85
|
+
self.modules = m
|
86
|
+
end
|
87
|
+
opts.on("-p", "--plugins-models", "Include plugins models") do |p|
|
88
|
+
self.plugins_models = p
|
89
|
+
end
|
90
|
+
opts.on("-t", "--transitive", "Include transitive associations",
|
91
|
+
"(through inheritance)") do |t|
|
92
|
+
self.transitive = t
|
93
|
+
end
|
94
|
+
opts.separator ""
|
95
|
+
opts.separator "Controllers diagram options:"
|
96
|
+
opts.on("--hide-public", "Hide public methods") do |h|
|
97
|
+
self.hide_public = h
|
98
|
+
end
|
99
|
+
opts.on("--hide-protected", "Hide protected methods") do |h|
|
100
|
+
self.hide_protected = h
|
101
|
+
end
|
102
|
+
opts.on("--hide-private", "Hide private methods") do |h|
|
103
|
+
self.hide_private = h
|
104
|
+
end
|
105
|
+
opts.separator ""
|
106
|
+
opts.separator "Other options:"
|
107
|
+
opts.on("-h", "--help", "Show this message") do
|
108
|
+
STDOUT.print "#{opts}\n"
|
109
|
+
exit
|
110
|
+
end
|
111
|
+
opts.on("--version", "Show version and copyright") do
|
112
|
+
STDOUT.print "#{APP_HUMAN_NAME} version #{APP_VERSION.join('.')}\n\n" +
|
113
|
+
"#{COPYRIGHT}\nThis is free software; see the source " +
|
114
|
+
"for copying conditions.\n\n"
|
115
|
+
exit
|
116
|
+
end
|
117
|
+
opts.separator ""
|
118
|
+
opts.separator "Commands (you must supply one of these):"
|
119
|
+
opts.on("-M", "--models", "Generate models diagram") do |c|
|
120
|
+
if self.command != ''
|
121
|
+
STDERR.print "Error: Can only generate one diagram type\n\n"
|
122
|
+
exit 1
|
123
|
+
else
|
124
|
+
self.command = 'models'
|
125
|
+
end
|
126
|
+
end
|
127
|
+
opts.on("-C", "--controllers", "Generate controllers diagram") do |c|
|
128
|
+
if self.command != ''
|
129
|
+
STDERR.print "Error: Can only generate one diagram type\n\n"
|
130
|
+
exit 1
|
131
|
+
else
|
132
|
+
self.command = 'controllers'
|
133
|
+
end
|
134
|
+
end
|
135
|
+
# From Ana Nelson's patch
|
136
|
+
opts.on("-A", "--aasm", "Generate \"acts as state machine\" diagram") do |c|
|
137
|
+
if self.command == 'controllers'
|
138
|
+
STDERR.print "Error: Can only generate one diagram type\n\n"
|
139
|
+
exit 1
|
140
|
+
else
|
141
|
+
self.command = 'aasm'
|
142
|
+
end
|
143
|
+
end
|
144
|
+
opts.separator ""
|
145
|
+
opts.separator "For bug reporting and additional information, please see:"
|
146
|
+
opts.separator "http://railroad.rubyforge.org/"
|
147
|
+
end # do
|
148
|
+
|
149
|
+
begin
|
150
|
+
@opt_parser.parse!(args)
|
151
|
+
rescue OptionParser::AmbiguousOption
|
152
|
+
option_error "Ambiguous option"
|
153
|
+
rescue OptionParser::InvalidOption
|
154
|
+
option_error "Invalid option"
|
155
|
+
rescue OptionParser::InvalidArgument
|
156
|
+
option_error "Invalid argument"
|
157
|
+
rescue OptionParser::MissingArgument
|
158
|
+
option_error "Missing argument"
|
159
|
+
end
|
160
|
+
end # parse
|
161
|
+
|
162
|
+
private
|
163
|
+
|
164
|
+
def option_error(msg)
|
165
|
+
STDERR.print "Error: #{msg}\n\n #{@opt_parser}\n"
|
166
|
+
exit 1
|
167
|
+
end
|
168
|
+
|
169
|
+
end # class OptionsStruct
|