Jifty-Plugin-AccessLog

 view release on metacpan or  search on metacpan

lib/Jifty/Plugin/AccessLog.pm  view on Meta::CPAN

package Jifty::Plugin::AccessLog;
use strict;
use warnings;
use base qw/Jifty::Plugin Class::Data::Inheritable/;
__PACKAGE__->mk_accessors(qw/path format start respect_proxy/);

use Jifty::Util;
use CGI::Cookie;
use Time::HiRes qw();

our $VERSION = 1.0;

=head1 NAME

Jifty::Plugin::AccessLog - Concisely log Jifty requests

=head1 DESCRIPTION


=head1 USAGE

Add the following to your site_config.yml

 framework:
   Plugins:
     - AccessLog: {}

=head2 OPTIONS

=over 4

=item path

The file to log to; defaults to F<log/access_log>.

=item respect_proxy

If set to a true value, will display the C<X-Forwarded-For> header as
the originating IP of requests.

=item format

The format string to use when logging.  This module attempts to be as
Apache-compatible as possible; it supports the following format
escapes:

=over

=item %%

The percent sign.

=item %a

Remote IP address.

=item %{Foobar}C

The contents of the cookie I<Foobar> in the request sent to the
server.

=item %D

The time taken to serve the request, in micoseconds.

=item %h

Remote IP address.

=item %{Foobar}n

The content of the I<Foobar> header line(s) in the request.

=item %l

The first 8 characters of the session ID, if any.

=item %m

The request method.

=item %{Foobar}n

The value of the template or request argument I<Foobar>, as sent by
the client, or set in the dispatcher.

=item %{Foobar}o

The value of the I<Foobar> header line(s) in the response.

=item %p

The canonical port of the server serving the request.  Alternate forms
include C<%{canonical}p>, C<%{local}p>, and C<%{remote}p>, which are
the respective connection ports.

=item %P

lib/Jifty/Plugin/AccessLog.pm  view on Meta::CPAN

    );

    return if $self->_pre_init;

    $self->path(Jifty::Util->absolute_path( $args{path} ));
    $self->format($args{format});
    $self->respect_proxy($args{respect_proxy});
    Jifty::Handler->add_trigger(
        before_cleanup => sub { $self->before_cleanup }
    );
}

=head2 new_request

On each request, log when it starts.

=cut

sub new_request {
    my $self = shift;
    $self->start(Time::HiRes::time);
}

=head2 before_cleanup

Open, and append to, the logfile with the format specified.

=cut

sub before_cleanup {
    my $self = shift;
    my $r    = Jifty->web->request;
    Jifty->web->response->status(200)
      unless defined Jifty->web->response->status;

    my $actions = sub {
        my $long = shift;

        my $one_action = sub {
            my $a = shift;
            my $base = $a->class;
            my $result = Jifty->web->response->result($a->moniker);
            $base .= "~" if not $a->has_run or not $result;
            $base .= "!" if $result and not $result->success;
            return $base unless $long;
            return "($base={"
                . join( ",",
                map { "$_=" . $a->argument($_) } keys %{ $a->arguments } )
                . "})"
        };
        return sub {
            my @a = grep { $_->active } $r->actions;
            return "-" unless @a;
            ( $r->just_validating ? "V" : "" ) . "<" . join(
                ", ",    map {$one_action->($_)} @a) . ">";
        }
    };

    my %ESCAPES = (
        '%' => sub { '%' },
        a => sub { ($self->respect_proxy && $r->header("X-Forwarded-For")) || $r->address },
        C => sub { my $c = { CGI::Cookie->fetch() }->{+shift}; $c ? $c->value : undef },
        D => sub { sprintf "%.3fms", (Time::HiRes::time - $self->start)*1000 },
        e => sub { $r->env->{+shift} },
        h => sub { ($self->respect_proxy && $r->header("X-Forwarded-For")) || $r->remote_host || $r->address },
        i => sub { $r->header(shift) },
        l => sub { substr( Jifty->web->session->id || '-', 0, 8 ) },
        m => sub { $r->method },
        n => sub { $r->template_argument($_[0]) || $r->argument($_[0]) },
        o => sub { Jifty->web->response->header(shift) },
        p => sub {
            return Jifty->config->framework("Web")->{Port} if $_[0] eq "canonical";
            return $r->env->{SERVER_PORT} if $_[0] eq "local";
            return $r->env->{REMOTE_PORT} if $_[0] eq "remote";
            return Jifty->config->framework("Web")->{Port};
        },
        P => sub { $$ },
        s => sub { Jifty->web->response->status =~ /^(\d+)/; $1 || "200" },
        t => sub { DateTime->from_epoch(epoch => $self->start)->strftime(shift || "[%d/%b/%Y:%T %z]") },
        T => sub { sprintf "%.3fs", (Time::HiRes::time - $self->start) },
        u => sub { Jifty->web->current_user->username },
        U => sub {
            if (my @f = $r->fragments) {
                return '[' . join(" ", map {s/ /%20/g;$_} map {$_->path} @f ) . ']';
            } else {
                my $path = $r->path;
                $path =~ s/ /%20/g;
                return $path;
            }
        },
        v => sub { URI->new(Jifty->config->framework("Web")->{BaseURL})->host },
        x => $actions->(0),
        X => $actions->(1),
    );

    my $replace = sub {
        my ($only_on, $string, $format) = @_;
        if (defined $only_on) {
            return "" unless grep {Jifty->web->response->status eq $_} split /,/, $only_on;
        }
        my $r;
        if (exists $ESCAPES{$format}) {
            $r = ref $ESCAPES{$format} ? eval {$ESCAPES{$format}->($string)} : $ESCAPES{$format};
        } else {
            $r = "%".$format;
        }
        return defined $r ? $r : "-";
    };


    my $s = $self->format;
    $s =~ s/%(\d+(?:,\d+)*)?(?:{(.*?)})?([a-zA-Z%])/$replace->($1,$2,$3)/ge;

    open my $access_log, '>>', $self->path or do {
        $self->log->error("Unable to open @{[$self->path]} for writing: $!");
        return;
    };
    $access_log->syswrite( "$s\n" );
    $access_log->close;
}

=head1 SEE ALSO

L<Jifty::Plugin::Recorder> for more verbose debugging information.



( run in 1.018 second using v1.01-cache-2.11-cpan-437f7b0c052 )