File: VF2.pm

package info (click to toggle)
libgraph-vf2-perl 0.2.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 104 kB
  • sloc: perl: 51; makefile: 3
file content (123 lines) | stat: -rw-r--r-- 4,055 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package Graph::VF2;

# ABSTRACT: VF2 subgraph isomorphism detection method for Perl Graph
our $VERSION = '0.2.0'; # VERSION

=head1 NAME

Graph::MoreUtils - VF2 subgraph isomorphism detection method for Perl Graph

=head1 SYNOPSIS

    use Graph::Undirected;
    use Graph::VF2 qw( matches );

    my $small = Graph::Undirected->new;
    my $large = Graph::Undirected->new;

    # Create graphs here

    # Find all subgraphs of $small in $large:
    my @matches = matches( $small, $large );

=cut

use strict;
use warnings;

use Graph::Undirected;

require Exporter;
our @ISA = qw( Exporter );
our @EXPORT_OK = qw(
    matches
);

require XSLoader;
XSLoader::load('Graph::VF2', $VERSION);

=head1 METHODS

=head2 C<matches( $g1, $g2, $options )>

Takes two L<Graph::Undirected> objects, C<$g1> and C<$g2> and returns an array of occurrences of C<$g1> in C<$g2>.
Returned array consists of array references, each array reference describing one occurrence.
In it, encoded as array references, is the list of pairwise vertex correspondences.
First item in a pair is a vertex from C<$g1>, and second item being a vertex in C<$g2>.
No attempt is made to collate isomorphic matches.
Thus a search of N-element cycle graph in itself will produce 2 * N matches due to graph's symmetry.

C<$options> is a hash reference of options with the following keys:

=over

=item C<vertex_correspondence_sub>

A subroutine reference used to evaluate the equality of vertices, called with C<$v1> and C<$v2> from C<$g1> and C<$g2>, accordingly.
Should return Perl true and false equivalents to signify match and non-match, accordingly.
Unless provided, all vertices are treated as equal.

=item C<edge_correspondence_sub>

A subroutine reference used to evaluate the equality of edges, called with C<$v1> and C<$v2> from C<$g1>, and C<$v3> and C<$v4> from C<$g2>, accordingly.
Edges are represented as pairs of vertices and are passed as a flat array.
Should return Perl true and false equivalents to signify match and non-match, accordingly.
Unless provided, all edges are treated as equal.

=back

=cut

sub matches
{
    my( $g1, $g2, $options ) = @_;

    die 'input graphs must be undirected'
        unless $g1->isa( Graph::Undirected:: ) && $g2->isa( Graph::Undirected:: );

    $options = {} unless $options;
    my $vertex_correspondence_sub = exists $options->{vertex_correspondence_sub}
                                         ? $options->{vertex_correspondence_sub}
                                         : sub { 1 };
    my $edge_correspondence_sub   = exists $options->{edge_correspondence_sub}
                                         ? $options->{edge_correspondence_sub}
                                         : sub { 1 };

    my @vertices1 = $g1->vertices;
    my %vertices1 = map { $vertices1[$_] => $_ } 0..$#vertices1;
    my @edges1    = map { [ $vertices1{$_->[0]}, $vertices1{$_->[1]} ] } $g1->edges;
    my @vertices2 = $g2->vertices;
    my %vertices2 = map { $vertices2[$_] => $_ } 0..$#vertices2;
    my @edges2    = map { [ $vertices2{$_->[0]}, $vertices2{$_->[1]} ] } $g2->edges;

    my $vertex_map = [];
    for my $vertex (@vertices2) {
        push @$vertex_map, [ map { int $vertex_correspondence_sub->($_, $vertex) } @vertices1 ];
    }
    my $edge_map = [];
    for my $edge (@edges2) {
        push @$edge_map, [ map { int $edge_correspondence_sub->( ( map { $vertices1[$_] } @$_ ),
                                                                 ( map { $vertices2[$_] } @$edge) ) } @edges1 ];
    }

    my $correspondence = _vf2( \@vertices1, \@edges1,
                               \@vertices2, \@edges2,
                               $vertex_map, $edge_map );

    my @matches;
    while (my @match = splice @$correspondence, 0, 2 * @vertices1) {
        push @matches, [ map { [ $vertices1[$match[2 * $_]],
                                 $vertices2[$match[2 * $_ + 1]] ] }
                             0..$#vertices1 ];
    }

    return @matches;
}

=head1 AUTHORS

Andrius Merkys, E<lt>merkys@cpan.orgE<gt>

=cut

1;