# frozen_string_literal: true class DiagramGraph attr_writer :diagram_type, :show_label, :alphabetize def initialize @diagram_type = '' @show_label = false @alphabetize = false @nodes = [] @edges = [] end def add_node(node) @nodes << node end def add_edge(edge) @edges << edge end # Generate DOT graph def to_dot dot_header + @nodes.map { |n| dot_node n[0], n[1], n[2], n[3] }.join + @edges.map { |e| dot_edge e[0], e[1], e[2], e[3] }.join + dot_footer end # Generate XMI diagram (not yet implemented) def to_xmi $stderr.print "Sorry. XMI output not yet implemented.\n\n" '' end private # Build DOT diagram header def dot_header result = "digraph #{@diagram_type.downcase}_diagram {\n" \ "\tgraph[overlap=false, splines=true, bgcolor=\"none\"]\n" result += dot_label if @show_label result end # Build DOT diagram footer def dot_footer "}\n" end # Build diagram label def dot_label "\t_diagram_info [shape=\"plaintext\", " \ "label=\"#{@diagram_type} diagram\\l" \ "Date: #{Time.now.strftime '%b %d %Y - %H:%M'}\\l" + (if defined?(ActiveRecord::Migrator) 'Migration version: ' \ "#{Rails.logger.silence { ActiveRecord::Migrator.current_version }}\\l" else '' end) + "Generated by #{APP_HUMAN_NAME} #{APP_VERSION}\\l" + 'http://railroady.prestonlee.com' \ "\\l\", fontsize=13]\n" end # Build a DOT graph node def dot_node(type, name, attributes = nil, custom_options = '') case type when 'model' options = "shape=Mrecord, label=\"{#{name}|" options += attributes.sort_by { |s| @alphabetize ? s : nil }.join('\l') options += '\l}"' when 'model-brief' options = '' when 'class' options = "shape=record, label=\"{#{name}|}\"" when 'class-brief' options = 'shape=box' when 'controller' options = "shape=Mrecord, label=\"{#{name}|" public_methods = attributes[:public].sort_by { |s| @alphabetize ? s : nil }.join('\l') protected_methods = attributes[:protected].sort_by { |s| @alphabetize ? s : nil }.join('\l') private_methods = attributes[:private].sort_by { |s| @alphabetize ? s : nil }.join('\l') options += "#{public_methods}\\l|#{protected_methods}\\l|#{private_methods}\\l" options += '}"' when 'controller-brief' options = '' when 'module' options = "shape=box, style=dotted, label=\"#{name}\"" when 'aasm' # Return subgraph format return "subgraph cluster_#{name.downcase.gsub(/[^a-z0-9\-_]+/i, '_')} {\n\tlabel = #{quote(name)}\n\t#{attributes.join("\n ")}}" end options = [options, custom_options].compact.reject(&:empty?).join(', ') "\t#{quote(name)} [#{options}]\n" end # Build a DOT graph edge def dot_edge(type, from, to, name = '') options = name != '' ? "label=\"#{name}\", " : '' edge_color = '"#%02X%02X%02X"' % [rand(255), rand(255), rand(255)] suffix = ", dir=both color=#{edge_color}" case type when 'one-one' options += "arrowtail=odot, arrowhead=dot#{suffix}" when 'one-many' options += "arrowtail=odot, arrowhead=crow#{suffix}" when 'many-many' options += "arrowtail=crow, arrowhead=crow#{suffix}" when 'belongs-to' # following http://guides.rubyonrails.org/association_basics.html#the-belongs-to-association options += "arrowtail=none, arrowhead=normal#{suffix}" when 'is-a' options += 'arrowhead="none", arrowtail="onormal"' when 'event' options += 'fontsize=10' end "\t#{quote(from)} -> #{quote(to)} [#{options}]\n" end # Quotes a class name def quote(name) "\"#{name}\"" end end