package SrvMngr::Controller::Qmailanalog; #---------------------------------------------------------------------- # heading : Investigation # description : Mail log file analysis # navigation : 7000 200 # # routes : end #---------------------------------------------------------------------- use strict; use warnings; use Mojo::Base 'Mojolicious::Controller'; use SrvMngr qw(gen_locale_date_string); use Locale::gettext; use SrvMngr::I18N; use SrvMngr qw(theme_list init_session); use List::Util qw(sum); #use Mail::Log::Trace::Postfix; sub main { my $c = shift; $c->app->log->info($c->log_req); my $title = $c->l('qma_FORM_TITLE'); my $modul = $c->render_to_string(inline => $c->l('qma_INITIAL_DESC')); $c->stash(title => $title, modul => $modul); $c->render(template => 'qmailanalog'); } ## end sub main sub do_update { my $c = shift; $c->app->log->info($c->log_req); my $result = ""; my $report_type = $c->param('report_type'); if ($report_type =~ /^(\S+)$/) { $report_type = $1; } elsif ($report_type =~ /^\s*$/) { $report_type = "zoverall"; } else { $result = $c->l('INVALID_REPORT_TYPE') . $report_type; $report_type = undef; } my $title = $c->l('qma_FORM_TITLE'); $result = $c->render_to_string(inline => generateReport($c, $report_type)) if $report_type; $c->stash(title => $title, modul => $result); $c->render(template => 'module'); } ## end sub do_update sub generateReport { my $c = shift; my $report_type = shift; my $out = ''; #------------------------------------------------------------ # Go ahead and generate the report. #------------------------------------------------------------ my $now_string = $c->gen_locale_date_string(); my $log_path = '/var/log/maillog'; $out .= sprintf("
"; # Get the selected report from the form submission my $selected_report = $report_type; # Call the relevant report sub based on the selection if ($selected_report eq 'daily_summary') { $out .= daily_summary_report($log_path); } elsif ($selected_report eq 'daily_summary_today') { $out .= daily_summary_report_today($log_path); } elsif ($selected_report eq 'daily_summary_all') { $out .= daily_summary_report_all($log_path); } elsif ($selected_report eq 'top_senders') { $out .= top_senders_and_recipients($log_path); } elsif ($selected_report eq 'bounce_analysis') { $out .= bounce_rate_analysis($log_path); } elsif ($selected_report eq 'spam_and_virus') { $out .= spam_and_virus_filter_report($log_path); } elsif ($selected_report eq 'delivery_status') { $out .= delivery_status_report($log_path); } elsif ($selected_report eq 'geo_analysis') { $out .= geographical_analysis_of_email($log_path); } elsif ($selected_report eq 'traffic_analysis') { $out .= traffic_analysis($log_path); } elsif ($selected_report eq 'auth_analysis') { $out .= authentication_analysis($log_path); } elsif ($selected_report eq 'user_activity') { $out .= user_activity_report($log_path); } elsif ($selected_report eq 'error_reporting') { $out .= error_reporting($log_path); } elsif ($selected_report eq 'comparison_reports') { $out .= comparison_reports($log_path, '/var/log/mail.log.1'); } elsif ($selected_report eq 'customized_reports') { $out .= customized_reports($log_path); } else { $out .= 'Invalid report selected'; } # The $output variable now contains the generated report output. # Further processing can be done here, or you can render it later. $out .= sprintf ""; $out .= sprintf("
$contentHTML } ### 1. Message Tracking #sub trace_message { #my ($log_path, $message_id) = @_; ##my $tracer = Mail::Log::Trace::Postfix->new({log_file => $log_path}); ##$tracer->set_message_id($message_id); #my $output = "Message Tracking Report for ID: $message_id\n"; ##$output .= "=" x 50 . "\n"; ##$output .= sprintf "%-12s: %s\n", 'From', $tracer->get_from_address; ##$output .= sprintf "%-12s: %s\n", 'Status', $tracer->get_final_status; ##$output .= "\nRecipients:\n"; ##$output .= join("\n", map { "- $_" } $tracer->get_recipient_addresses); ##$output .= "\n\nTimeline:\n"; ##my $timeline = $tracer->get_timestamps; ##while (my ($stage, $time) = each %$timeline) { ##$output .= sprintf "%-10s: %s\n", ucfirst($stage), $time; ##} #return $output || "No records found for message ID: $message_id"; #} #### 2. Queue Analysis #sub get_queue_stats { #my $spool_dir = '/var/spool/postfix'; #my %queues = map { $_ => 0 } qw(active deferred bounce hold corrupt); #foreach my $q (keys %queues) { #opendir(my $dh, "$spool_dir/$q"); #$queues{$q} = scalar(grep { -f "$spool_dir/$q/$_" } readdir($dh)); #closedir($dh); #} #my $output = "Current Postfix Queue Status\n"; #$output .= "=" x 30 . "\n"; #$output .= sprintf "%-10s: %3d messages\n", ucfirst($_), $queues{$_} #for sort keys %queues; #$output .= "\nTotal: " . sum(values %queues) . " messages in queue"; #return $output; #} #### 3. Message Statistics #sub get_message_stats { #my ($log_path) = @_; #my %stats = (received => 0, rejected => 0, delivered => 0, #deferred => 0, bounced => 0, held => 0); #open(my $fh, '<', $log_path); #while(<$fh>) { #$stats{received}++ if /qmgr.*: [A-Z0-9]+: from=/; #$stats{delivered}++ if /status=sent/; #$stats{rejected}++ if /NOQUEUE: reject/; #$stats{deferred}++ if /status=deferred/; #$stats{bounced}++ if /status=bounced/; #$stats{held}++ if /status=hold/; #} #close($fh); #my $output = "Message Statistics for " . localtime . "\n"; #$output .= "=" x 40 . "\n"; #$output .= sprintf "%-12s: %6d\n", 'Received', $stats{received}; #$output .= sprintf "%-12s: %6d (%.1f%%)\n", 'Delivered', $stats{delivered}, #($stats{received} ? ($stats{delivered}/$stats{received}*100) : 0); #$output .= sprintf "%-12s: %6d\n", 'Rejected', $stats{rejected}; #$output .= sprintf "%-12s: %6d\n", 'Deferred', $stats{deferred}; #$output .= sprintf "%-12s: %6d\n", 'Bounced', $stats{bounced}; #$output .= sprintf "%-12s: %6d\n", 'Held', $stats{held}; #return $output; #} #### 4. User Activity Audit #sub get_user_activity { #my ($log_path, $email) = @_; ##my $tracer = Mail::Log::Trace::Postfix->new({log_file => $log_path}); ##my $sent = scalar $tracer->find_messages_by_sender($email); ##my $received = scalar $tracer->find_messages_by_recipient($email); #my $output = "Activity Report for: $email\n"; ##$output .= "=" x (length($email) + 18) . "\n"; ##$output .= "Messages sent: $sent\n"; ##$output .= "Messages received: $received\n\n"; ##$output .= "Last week's activity:\n"; ##$output .= join("\n", map { sprintf "- %s: %d messages", $_->[0], $_->[1] } ##$tracer->get_weekly_stats($email)); #return $output || "No activity found for $email"; #} #### 5. Security Monitoring #sub detect_auth_failures { #my ($log_path) = @_; #my %failures; #open(my $fh, '<', $log_path); #while(<$fh>) { #if(/SASL (?:LOGIN|PLAIN) authentication failed.*?\[([0-9.]+)\]/) { #$failures{$1}++; #} #} #close($fh); #return "No authentication failures found" unless keys %failures; #my $output = "Authentication Failure Report\n"; #$output .= "=" x 30 . "\n"; #$output .= sprintf "%-15s %s\n", 'IP Address', 'Attempts'; #$output .= sprintf "%-15s %s\n", '-' x 15, '-' x 7; #foreach my $ip (sort { $failures{$b} <=> $failures{$a} } keys %failures) { #$output .= sprintf "%-15s %5d\n", $ip, $failures{$ip}; #} #$output .= "\nTotal failures: " . sum(values %failures); #return $output; #} 1;