Group
Extension

Stepford-Grapher/lib/Stepford/Grapher/Renderer/Graphviz.pm

package Stepford::Grapher::Renderer::Graphviz;

use strict;
use warnings;
use namespace::autoclean;
use autodie qw(:all);

use Moose;

use File::Temp qw( tempfile );
use GraphViz2;
use Stepford::Grapher::Types qw( HashRef Str );

our $VERSION = '1.01';

has output => (
    is        => 'ro',
    isa       => Str,
    predicate => 'has_output',
);

has format => (
    is      => 'ro',
    isa     => Str,
    lazy    => 1,
    builder => '_build_format',
);

sub _build_format {
    my $self = shift;

    # attempt to use the file extension as the format, if there is a usable
    # extension that is...
    my $ext;
    unless ( $self->has_output
        && ( ($ext) = $self->output =~ /[.]([^.]+)\z/ ) ) {
        return 'src';
    }

    return $ext;
}

# TODO: Make this configurable from the command line, either by accepting some
# sort of JSON-as-command-line-argument-flag setting, or by having multiple
# attributes that *are* individually settable and are lazily built into this
# formatting hashref if nothing is passed.
has node_formatting => (
    is      => 'ro',
    isa     => HashRef,
    builder => '_build_node_formatting',
);

sub _build_node_formatting {
    return {
        fontname => 'Helvetica',
        fontsize => 9,
        shape    => 'rect',
    };
}

has edge_formatting => (
    is      => 'ro',
    isa     => HashRef,
    builder => '_build_edge_formatting',
);

sub _build_edge_formatting {
    return {
        fontname => 'Helvetica',
        fontsize => 7,
    };
}

with 'Stepford::Grapher::Role::Renderer';

sub _create_blank_graph {
    return GraphViz2->new;
}

sub _populate_graph {
    my $self  = shift;
    my $graph = shift;
    my $data  = shift;

    $graph->add_node(
        name => $_,
        %{ $self->node_formatting }
    ) for keys %{$data};

    for my $from ( keys %{$data} ) {
        for my $label ( keys %{ $data->{$from} } ) {
            $graph->add_edge(
                from  => $from,
                to    => $data->{$from}{$label},
                label => $label,
                %{ $self->edge_formatting }
            );
        }
    }

    return;
}

sub _output_graph {
    my $self  = shift;
    my $graph = shift;

    # are we rendering to a named file or a temp file?
    my $output = (
        $self->has_output ? $self->output : do {
            my ( undef, $filename ) = tempfile();
            $filename;
            }
    );

    $graph->run(
        format => ( $self->format eq 'src' ? 'dot' : $self->format ),
        output_file => $output,
    );

    # If we were rendering to STDOUT, send to STDOUT
    unless ( $self->has_output ) {
        open my $fh, '<:raw', $output;
        while (<$fh>) {
            print or die "Can't print: $!";
        }
        close $fh;
        unlink $output;
    }

    return;
}

sub render {
    my $self = shift;
    my $data = shift;

    my $graph = $self->_create_blank_graph;
    $self->_populate_graph( $graph, $data );
    $self->_output_graph($graph);
}

__PACKAGE__->meta->make_immutable;
1;

=pod

=encoding UTF-8

=head1 NAME

Stepford::Grapher::Renderer::Graphviz - Render to a graph using GraphViz

=head1 VERSION

version 1.01

=head1 SYNOPSIS

   my $grapher = Stepford::Grapher->new(
       step  => 'My::Step::ExampleStep',
       step_namespaces => ['My::Steps'],
       renderer => Stepford::Grapher::Renderer::Graphiz->new(
           output => 'diagram.png',
       ),
   );
   $grapher->run;

=head1 DESCRIPTION

Renders the graph using GraphViz.

=head1 ATTRIBUTES

=head2 output

A string containing the filename that the rendered graph should be written to.
By default this is undef, rendering the output to C<STDOUT>.

=head2 format

The format the graph should be written in (C<jpg>, C<png>, C<pdf>, etc.)  You
may use the special format C<src> or C<dot> to indicate that you want simple dot
source code (for manually feeding into Graphviz) output rather than having the
rendering done for you.

By default the format is determined from the output file extension if there
is one.  If one cannot be determined (either because the C<output> attribute
is undefined or the filename does not have an extension) then format will
default to C<src>.

=head1 METHOD

=head2 $renderer->render()

Renders the output.

=head1 SUPPORT

Bugs may be submitted through L<https://github.com/maxmind/Stepford-Grapher/issues>.

=head1 AUTHOR

Mark Fowler <mfowler@maxmind.com>

=head1 COPYRIGHT AND LICENSE

This software is copyright (c) 2016 - 2017 by MaxMind, Inc..

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.

=cut

__END__

# ABSTRACT: Render to a graph using GraphViz



Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.