App-Chronicle

 view release on metacpan or  search on metacpan

lib/Chronicle/Plugin/Generate/Archive.pm  view on Meta::CPAN


package Chronicle::Plugin::Generate::Archive;


use strict;
use warnings;
use Date::Language;
use Encode qw/ decode /;

our $VERSION = "5.1.7";

=head2 on_generate

The C<on_generate> method is automatically invoked to generate output
pages.  This particular plugin method is invoked I<after> any
C<on_initiate> methods which might be present.

This method is responsible for generating the archive-output, which
includes two sets of pages:

=over 8

=item C</archive/index.html>

This is created using the C<archive_index.tmpl> theme-template, and contains
a list of all the year/month pairs which have blog-posts present for them.

=item C</archive/$year/$mon/index.html>

This is created for each distinct year/month pair, from the theme-template
C<archive.tmpl>

=back

If either template is missing then this plugin will skip that part of
the generation.

=cut

sub on_generate
{
    my ( $self, %args ) = (@_);

    my $dbh    = $args{ 'dbh' };
    my $config = $args{ 'config' };

    #
    #  Date-record
    #
    my %hash;
    my %index;

    my $all = $dbh->prepare(
         "SELECT strftime( '%m %Y', date, 'unixepoch') FROM blog ORDER BY date")
      or
      die "Failed to prepare";


    $all->execute() or die "Failed to execute:" . $dbh->errstr();
    my $dt;
    $all->bind_columns( undef, \$dt );

    while ( $all->fetch() )
    {
        if ( $dt =~ /([0-9]+) ([0-9]+)/ )
        {
            $hash{ $dt } += 1;

            $index{ $2 }{ $1 } += 1;
        }
    }
    $all->finish();

    #
    #  Ouptut the index
    #
    my $data;
    foreach my $year ( reverse sort keys %index )
    {
        my $mons = $index{ $year };

        foreach my $mon ( reverse sort keys %$mons )
        {
            push @$data,
              { year  => $year,
                month => $mon,
                count => $index{ $year }{ $mon } };
        }
    }


    if ( !-d "$config->{'output'}/archive/" )
    {
        File::Path::make_path( "$config->{'output'}/archive/",
                               {  verbose => 0,
                                  mode    => oct("755"),
                               } );
    }

    #
    #  The index file to generate
    #
    my $index = $config->{ 'index_filename' } || "index.html";

    my $c = Chronicle::load_template("archive_index.tmpl");
    if ($c)
    {
        my $index_path = "$config->{'output'}/archive/$index";
        print "Creating : $index_path\n" if $config->{ 'verbose' };
        $c->param( top => $config->{ 'top' } );
        $c->param( archive => $data ) if ($data);
        open my $handle, ">:encoding(UTF-8)", $index_path or
          die "Failed to open `$index_path': $!";
        print $handle $c->output();
        close $handle;
    }



    #
    #  Foreach year/mon pair
    #
    foreach my $ym ( keys %hash )
    {
        my $datelang = Date::Language->new( $ENV{ 'MONTHS' } // "English" );
        my $mon      = "";
        my $year     = "";
        if ( $ym =~ /^([0-9]+) ([0-9]+)$/ )
        {
            $mon  = $1;
            $year = $2;
        }
        my $ids = $dbh->prepare(
            "SELECT id FROM blog WHERE strftime( '%m %Y', date, 'unixepoch') = ? ORDER BY date DESC"
          ) or
          die "Failed to prepare";

        $ids->execute($ym) or die "Failed to execute:" . $dbh->errstr();

        my $id;
        $ids->bind_columns( undef, \$id );

        my $ym_archive_path = "$config->{'output'}/archive/$year/$mon";

        # Make path unless it exists
        File::Path::make_path( $ym_archive_path,
                               {  verbose => 0,
                                  mode    => 0755,
                               } )
          unless -e $ym_archive_path;

        my $entries;

        while ( $ids->fetch() )
        {
            my $post =
              Chronicle::getBlog( dbh    => $dbh,
                                  id     => $id,
                                  config => $config
                                );
            if ( $config->{ 'lower-case' } )
            {
                $post->{ 'link' } = lc( $post->{ 'link' } );
            }

            push( @$entries, $post );

        }
        $ids->finish();

        my $ym_index_path = "$ym_archive_path/$index";
        print "Creating : $ym_index_path\n" if $config->{ 'verbose' };

        $c = Chronicle::load_template("/archive.tmpl");
        return if ( !$c );

        $c->param( top     => $config->{ 'top' } );
        $c->param( entries => $entries );
        $c->param( month   => $mon );
        $c->param(
                month_name => decode(
                    'ISO-8859-1', $datelang->time2str( '%B', 28 * 86400 * $mon )
                                    ) );
        $c->param( year => $year );
        open my $handle, ">:encoding(UTF-8)", $ym_index_path or
          die "Failed to open `$ym_index_path': $!";
        print $handle $c->output();
        close $handle;
    }
}


1;


=head1 LICENSE

This module is free software; you can redistribute it and/or modify it
under the terms of either:

a) the GNU General Public License as published by the Free Software



( run in 1.221 second using v1.01-cache-2.11-cpan-2398b32b56e )