package UML::Class::Simple;
use strict;
use warnings;
no warnings 'redefine';
use Class::Inspector;
use IPC::Run3;
use Template;
use Carp qw(carp);
use File::Spec;
use List::MoreUtils 'any';
require Exporter;
our @ISA = qw(Exporter);
our @EXPORT = qw(
classes_from_runtime classes_from_files
exclude_by_paths grep_by_paths
);
our $VERSION = '0.09';
my $tt = Template->new;
my $dot_template;
sub classes_from_runtime {
my ($modules, $pattern) = @_;
$modules = [$modules] if $modules and !ref $modules;
$pattern = '' if !defined $pattern;
for (@$modules) {
eval "use $_;";
if ($@) { carp $@; return (); }
}
grep { /$pattern/ } _runtime_packages();
}
sub _normalize_path ($) {
my $path = shift;
$path = File::Spec->rel2abs($path);
if (File::Spec->case_tolerant()) {
$path = lc($path);
} else {
$path;
}
}
sub exclude_by_paths ($@) {
my $rclasses = shift;
my @paths = map { _normalize_path($_) } @_;
my @res;
#_extend_INC();
for my $class (@$rclasses) {
#warn $class;
my $filename = Class::Inspector->resolved_filename($class);
#warn "[0] ", $filename, "\n";
if (!$filename && $INC{$class}) {
$filename = Class::Inspector->loaded_filename($class);
}
if (!$filename) { next; }
#warn "[1] ", $filename, "\n";
$filename = _normalize_path($filename);
#warn "[2] ", $filename, "\n";
#my $value = $INC{$key};
if (any { substr($filename, 0, length) eq $_ } @paths) {
#warn "!!! ignoring $filename\n";
next;
}
#warn "adding $filename <=> @paths\n";
push @res, $class;
}
@res;
}
sub grep_by_paths ($@) {
my $rclasses = shift;
my @paths = map { _normalize_path($_) } @_;
my @res;
#_extend_INC();
for my $class (@$rclasses) {
my $filename = Class::Inspector->resolved_filename($class);
if (!$filename && $INC{$class}) {
$filename = Class::Inspector->loaded_filename($class);
}
if (!$filename) { next; }
$filename = _normalize_path($filename);
#my $value = $INC{$key};
if (any { substr($filename, 0, length) eq $_ } @paths) {
#warn "adding $filename <=> @paths\n";
push @res, $class;
next;
}
#warn "!!! ignoring $filename\n";
}
@res;
}
sub _runtime_packages {
no strict 'refs';
my $pkg_name = shift || '::';
my $cache = shift || {};
return if $cache->{$pkg_name};
$cache->{$pkg_name} = 1;
for my $entry (keys %$pkg_name) {
next if $entry !~ /\:\:$/ or $entry eq 'main::';
my $subpkg_name = $pkg_name.$entry;
#warn $subpkg_name;
_runtime_packages($subpkg_name, $cache);
$cache->{$subpkg_name} = 1;
}
map { s/^::|::$//g; $_ } keys %$cache;
}
sub classes_from_files {
require PPI;
my ($list, $pattern, $read_only) = @_;
$list = [$list] if $list and !ref $list;
$pattern = '' if !defined $pattern;
my @classes;
my $cache = {};
for my $file (@$list) {
_gen_paths($file, $cache);
my $doc = PPI::Document->new( $file );
if (!$doc) {
carp "warning: Can't parse $file: ", PPI::Document->errstr;
next;
}
my $res = $doc->find('PPI::Statement::Package');
next if !$res;
push @classes, map { $_->namespace } @$res;
_load_file($file) if !$read_only;
}
@classes = grep { /$pattern/ } @classes;
#@classes = sort @classes;
wantarray ? @classes : \@classes;
}
sub _gen_paths {
my ($file, $cache) = @_;
$file =~ s{\\+}{/}g;
my $dir;
while ($file =~ m{(?x) \G .+? /+ }gc) {
$dir .= $&;
next if $cache->{$dir};
$cache->{$dir} = 1;
#warn "pushing ~~~ $dir\n";
unshift @INC, $dir;
}
}
sub new {
my $class = ref $_[0] ? ref shift : shift;
my $rclasses = shift || [];
my $self = bless {
class_names => $rclasses,
node_color => '#f1e1f4',
}, $class;
$self->_build_dom;
$self;
}
sub size {
my $self = shift;
if (@_) {
my ($width, $height) = @_;
if (!$width || !$height || ($width . $height) !~ /^[\.\d]+$/) {
carp "invalid width and height";
return undef;
} else {
$self->{width} = $width;
$self->{height} = $height;
return 1;
}
} else {
return ($self->{width}, $self->{height});
}
}
sub node_color {
my $self = shift;
if (@_) {
$self->{node_color} = shift;
} else {
$self->{node_color};
}
}
sub public_only {
my $self = shift;
if (@_) {
$self->{public_only} = shift;
$self->_build_dom(1);
} else {
$self->{public_only};
}
}
sub as_png {
my $self = shift;
$self->_as_image('png', @_);
}
sub as_gif {
my $self = shift;
$self->_as_image('gif', @_);
}
sub _as_image {
my ($self, $type, $fname) = @_;
my $dot = $self->as_dot;
#if ($fname eq 'fast00.png') {
#warn "==== $fname\n";
#warn $dot;
#use YAML::Syck;
#$self->_build_dom(1);
#warn Dump($self->as_dom);
#}
my @cmd = ('dot', '-T', $type);
if ($fname) {
push @cmd, '-o', $fname;
}
my ($img_data, $stderr);
my $success = run3 \@cmd, \$dot, \$img_data, \$stderr;
if ($stderr) {
carp $stderr;
}
if (!$fname) {
return $img_data;
}
}
sub as_dom {
my $self = shift;
$self->_build_dom;
{ classes => $self->{classes} };
}
sub set_dom ($$) {
my $self = shift;
$self->{classes} = shift->{classes};
1;
}
sub _build_dom {
my ($self, $force) = @_;
# avoid unnecessary evaluation:
return if $self->{classes} && !$force || !$self->{class_names};
#warn "HERE";
my @pkg = @{ $self->{class_names} };
my @classes;
$self->{classes} = \@classes;
my $public_only = $self->{public_only};
my %visited; # used to eliminate potential repetitions
for my $pkg (@pkg) {
#warn $pkg;
$pkg =~ s/::::/::/g;
if ($visited{$pkg}) { next; }
$visited{$pkg} = 1;
if (!Class::Inspector->loaded($pkg)) {
#my $pmfile = Class::Inspector->filename($pkg);
#warn $pmfile;
#if ($pmfile) {
# if (! _load_file($pmfile)) {
# next;
# }
#} else { next }
next;
}
push @classes, {
name => $pkg, methods => [],
properties => [], subclasses => [],
};
my $func = Class::Inspector->functions($pkg);
if ($func and @$func) {
if ($public_only) {
@$func = grep { /^[^_]/ } @$func;
}
$classes[-1]->{methods} = $func;
}
my $subclasses = Class::Inspector->subclasses($pkg);
if ($subclasses) {
no strict 'refs';
my @child = grep {
#warn "!!!! ", join ' ', @{"${_}::ISA"};
any { $_ eq $pkg } @{"${_}::ISA"};
} @$subclasses;
if (@child) {
$classes[-1]->{subclasses} = \@child;
}
}
}
#warn "@classes";
}
sub _load_file ($) {
my $file = shift;
my $path = _normalize_path($file);
#warn "!!! >>>> $path\n";
if ( any {
#warn "<<<<< ", _normalize_path($_), "\n";
$path eq _normalize_path($_);
} values %INC ) {
#carp "!!! Caught duplicate module files: $file ($path)";
return 1;
}
#my @a = values %INC;
#warn "\n@a\n";
#warn "!!! Loading $path...\n";
eval {
require $path;
};
carp $@ if $@;
!$@;
}
sub as_dot {
my ($self, $fname) = @_;
$self->_build_dom;
if ($fname) {
$tt->process(\$dot_template, $self, $fname)
|| carp $tt->error();
} else {
my $dot;
$tt->process(\$dot_template, $self, \$dot)
|| carp $tt->error();
$dot;
}
}
sub set_dot ($$) {
my $self = shift;
$self->{dot} = shift;
}
$dot_template = <<'_EOC_';
digraph uml_class_diagram {
[%- IF width && height %]
size="[% width %],[% height %]";
[%- END %]
node [shape=record, style="filled"];
edge [color=red, dir=none];
[%- name2id = {} %]
[%- id = 1 %]
[%- FOREACH class = classes %]
[%- name = class.name %]
[%- name2id.$name = id %]
class_[% id %] [shape=plaintext, style="", label=<
| [% name %] |
[% IF class.properties.size > 0 %]
[%- FOREACH property = class.properties %]
[%- property.match("^_") ? "-" : "+" %]
[%- END %][% END %] |
[%- FOREACH property = class.properties %]
[%- property %]
[%- END %] |
|
[% IF class.methods.size > 0 %]
[%- FOREACH method = class.methods %]
[%- method.match("^_") ? "-" : "+" %]
[%- END %][% END %] |
[%- FOREACH method = class.methods %]
[%- method %]
[%- END %] |
|
>];
[%- id = id + 1 %]
[% END %]
[%- class_id = id %]
[%- first = 1 %]
[%- id = 0 %]
[%- FOREACH class = classes %]
[%- id = id + 1 %]
[%- super = class.name %]
[%- NEXT IF !class.subclasses.size -%]
[%- IF first -%]
node [shape="triangle", fillcolor=yellow, height=0.3, width=0.3];
[%- first = 0 %]
[%- END -%]
angle_[% id %] [label=""];
[%- super_id = name2id.$super %]
class_[% super_id %]:methods -> angle_[% id %]
[%- FOREACH child = class.subclasses %]
[%- child_id = name2id.$child %]
[%- IF !child_id %]
class_[% class_id %] [shape=record, label="[% child %]" fillcolor="#f1e1f4", style="filled"];
angle_[% id %] -> class_[% class_id %]
[%- class_id = class_id + 1 %]
[%- ELSE %]
angle_[% id %] -> class_[% child_id %]:title
[%- END %]
[%- END %]
[%- END %]
}
_EOC_
1;
__END__
=head1 NAME
UML::Class::Simple - Render simple UML class diagrams, by loading the code
=head1 VERSION
This document describes C 0.09 released by April 10, 2007.
=head1 SYNOPSIS
use UML::Class::Simple;
# produce a class diagram for Alias's PPI
# which has already installed to your perl:
@classes = classes_from_runtime("PPI", qr/^PPI::/);
$painter = UML::Class::Simple->new(\@classes);
$painter->as_png('ppi.png');
# produce a class diagram for your CPAN module on the disk
@classes = classes_from_files(['lib/Foo.pm', 'lib/Foo/Bar.pm']);
$painter = UML::Class::Simple->new(\@classes);
# we can explicitly specify the image size
$painter->size(5, 3.6); # in inches
# ...and change the default title background color:
$painter->node_color('#ffffff'); # defaults to '#f1e1f4'
# only show public methods and properties
$painter->public_only(1);
$painter->as_png('my_module.png');
=head1 DESCRIPTION
C is a Perl CPAN module that generates UML class
diagrams (PNG format, GIF format, or dot source) automatically
from Perl 5 source or Perl 5 runtime.
Perl developers can use this module to obtain pretty class diagrams
for arbitrary existing Perl class libraries (including modern perl OO
modules based on Moose.pm), by only a single command. Companies can
also use the resulting pictures to visualize the project hierarchy and
embed them into their documentation.
The users no longer need to drag a mouse on the screen so as to draw
figures themselves or provide any specs other than the source code of
their own libraries that they want to depict. This module does all the
jobs for them! :)
You know, I was really impressed by the outputs of L, so I
decided to find something to (automatically) get pretty class diagrams
too. The images from L's Graphviz backend didn't quite fit my needs
when I was making some slides for my presentations.
I think most of the time you just want to use the command-line utility
L offered by this module (just like me). See the
documentation of L for details.
=head1 SAMPLE OUTPUTS
=over
=item PPI
L
=begin html
=end html
(See also F in the distribution.)
=item Moose
L
=begin html
=end html
(See also F in the distribution.)
=item FAST
L
=begin html
=end html
(See also F in the distribution.)
=back
=head1 SUBROUTINES
=over
=item classes_from_runtime($module_to_load, $regex?)
=item classes_from_runtime(\@modules_to_load, $regex?)
Returns a list of class (or package) names by inspecting the perl runtime environment.
C<$module_to_load> is the I module name to load while C<$regex> is
a perl regex used to filter out interesting package names.
The second argument can be omitted.
=item classes_from_files($pmfile, $regex?)
=item classes_from_files(\@pmfiles, $regex?)
Returns a list of class (or package) names by scanning through the perl source files
given in the first argument. C<$regex> is used to filter out interesting package names.
The second argument can be omitted.
=item exclude_by_paths
Excludes package names via specifying one or more paths where the corresponding
modules were installed into. For example:
@classes = exclude_by_paths(\@classes, 'C:/perl/lib');
@classes = exclude_by_paths(\@classes, '/home/foo', '/System/Library');
=item grep_by_paths
Filters out package names via specifying one or more paths where the corresponding
modules were installed into. For instance:
@classes = grep_by_paths(\@classes, '/home/malon', './blib/lib');
=back
All these subroutines are exported by default.
=head1 METHODS
=over
=item C<< $obj->new( [@class_names] ) >>
Create a new C instance with the specified class name list.
This list can either be constructed manually or by the utility functions
C and C.
=item C<< $obj->as_png($filename?) >>
Generate PNG image file when C<$filename> is given. It returns
binary data when C<$filename> is not given.
=item C<< $obj->as_gif($filename?) >>
Similar to C, bug generate a GIF-format image.
=item C<< $obj->as_dom() >>
Return the internal DOM tree used to generate dot and png. The tree's structure
looks like this:
{
'classes' => [
{
'subclasses' => [],
'methods' => [],
'name' => 'PPI::Structure::List',
'properties' => []
},
{
'subclasses' => [
'PPI::Structure::Block',
'PPI::Structure::Condition',
'PPI::Structure::Constructor',
'PPI::Structure::ForLoop',
'PPI::Structure::Unknown'
],
'methods' => [
'_INSTANCE',
'_set_finish',
'braces',
'content',
'new',
'refaddr',
'start',
'tokens'
],
'name' => 'PPI::Structure',
'properties' => []
},
...
]
}
You can adjust the data structure and feed it back to C<$obj> via
the C method.
=item C<< $obj->set_dom($dom) >>
Set the internal DOM structure to C<$obj>. This will be used to
generate the dot source and thus the PNG/GIF images.
=item C<< $obj->as_dot() >>
Return the Graphviz dot source code generated by C<$obj>.
=item C<< $obj->set_dot($dot) >>
Set the dot source code used by C<$obj>.
=back
=head1 PROPERTIES
=over
=item C<< $obj->size($width, $height) >>
=item C<< ($width, $height) = $obj->size >>
Set/get the size of the output images, in inches.
=item C<< $obj->public_only($bool) >>
=item C<< $bool = $obj->public_only >>
When the C property is set to true, only public methods or properties
are shown. It defaults to false.
=item C<< $obj->node_color($color) >>
=item C<< $color = $obj->node_color >>
Set/get the background color for the class nodes. It defaults to C<'#f1e1f4'>.
=back
=head1 INSTALLATION
Please download and intall a recent Graphviz release from its home:
L
C requires the HTML label feature which is only
available on versions of Graphviz that are newer than mid-November 2003.
In particular, it is not part of release 1.10.
Add Graphviz's F path to your PATH environment. This module needs its
F utility.
Grab this module from the CPAN mirror near you and run the following commands:
perl Makefile.PL
make
make test
make install
For windows users, use C instead of C.
Note that it's recommended to use the C utility to install CPAN modules.
=head1 LIMITATIONS
=over
=item *
It's pretty hard to distinguish perl methods from properties (actually they're both
implemented by subs in perl). If you have any good thoughts on this issue,
please drop me a line.
=item *
Only the inheritance relationships are shown in the images. I believe
other subtle
relations may mess up the Graphviz layouter. Hence the "::Simple" suffix in
this module name.
=item *
Unlike L, at this moment only Graphviz backend is provided.
=item *
There's no way to recognize I perl classes automatically. After all, Perl 5's
classes are implemented by packages. I think Perl 6 will make my life much easier.
=item *
To prevent potential naming confusion. I'm using Perl's C<::> namespace
separator
in the class diagrams instead of dot (C<.>) chosen by the UML standard.
One can argue that following UML standards is more important since people
in the same team may
use different programming languages, but I think it's not the case for
the majority (including myself) ;-)
=back
=head1 TODO
=over
=item *
Add more unit tests.
=item *
Add support for more image formats, such as as_ps, as_jpg, and etc.
=item *
Plot class relationships other than inheritance on the user's request.
=item *
Provide backends other than Graphviz.
=back
Please send me your wish list by emails or preferably via the CPAN RT site.
I'll add them here or even implement them promptly if I'm also interested
in your (crazy) ideas. ;-)
=head1 BUGS
There must be some serious bugs lurking somewhere;
if you found one, please report
it to L or contact the author directly.
=head1 ACKNOWLEDGEMENT
I must thank Adam Kennedy (Alias) for writing the excellent L and
L modules. L uses the former to extract
package names
from user's F<.pm> files or the latter to retrieve the function list of a
specific package.
I'm also grateful to Christopher Malon since he has (unintentionally)
motivated me to turn the original hack into this CPAN module. ;-)
=head1 SOURCE CONTROL
You can always grab the latest version from the following Subversion
repository:
L
It has anonymous access to all.
If you have the tuits to help out with this module, please let me know.
I have a dream to keep sending out commit bits like Audrey Tang. ;-)
=head1 AUTHOR
Agent Zhang Eagentzh@gmail.comE
=head1 COPYRIGHT
Copyright 2006 by Agent Zhang. All rights reserved.
This library is free software; you can redistribute it and/or modify it under
the same terms as perl itself.
=head1 SEE ALSO
L, L, L, L, L.