Skip to content
Snippets Groups Projects
Commit b1ee36be authored by Hallvard Furuseth's avatar Hallvard Furuseth
Browse files

Add slapd-tools/statslog, useful to search and inspect statslog output.

parent 55b40211
Branches
Tags
No related merge requests found
......@@ -22,6 +22,9 @@ Current contributions:
slapd-modules
Native modules
slapd-tools
Tools to use with slapd
slapi-plugins
SLAPI plugins
......
Copyright 2004 The OpenLDAP Foundation. All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted only as authorized by the OpenLDAP
Public License.
Directory contents:
statslog
Program to output selected parts of slapd's statslog output
(LDAP request/response log), grouping log lines by LDAP
connection. Useful to search and inspect the server log.
#!/usr/bin/perl -w
# statslog - Rearrange and output selected parts of slapd's statslog output.
# $OpenLDAP$
#
# Copyright 2004 Hallvard B. Furuseth.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted only as authorized by the OpenLDAP
# Public License.
#
# A copy of this license is available in the file LICENSE in the
# top-level directory of the distribution or, alternatively, at
# <http://www.OpenLDAP.org/license.html>.
sub usage {
die join("", @_, <<'EOM');
Usage: statslog [options] [logfiles; may be .gz or .bz2 files]
Output selected parts of slapd's statslog output (LDAP request/response
log to syslog or stderr; loglevel 256), grouping log lines by LDAP
connection. Lines with no connection are excluded by default.
Options:
--brief -b Brief output (omit time, host/process name/ID).
--exclude=RE -e RE Exclude connections whose output matches REgexp.
--include=RE -i RE Only include connections matching REgexp.
--EXCLUDE=RE -E RE Case-sensitive '--exclude'.
--INCLUDE=RE -I RE Case-sensitive '--include'.
--loose -l Include "loose" lines (lines with no connection).
--no-loose -L RE Only exclude the "loose" lines that match RE.
--join -j Join the inputs as if they were one big log file.
Each file must start where the previous left off.
--no-join -J Do not --join. (Can be useful with --sort.)
--sort -s Sort input files by age. Implies --join.
--trace -t Print file names when read. Implies --no-join.
All --exclude/include options are applied. Note: --exclude/include are
unreliable without --join/sort for connections spanning several log files.
EOM
}
########################################################################
use bytes;
use strict;
use Getopt::Long;
# Globals
my %conns; # Hash (connection number -> output)
my @loose; # Collected output with no connection number
# Command line options
my($brief, @filters, @conditions, $no_loose);
my($join_files, $sort_files, $trace, $getopt_ok);
# Handle --include/INCLUDE/exclude/EXCLUDE options
sub filter_opt {
my($opt, $regexp) = @_;
push(@conditions, sprintf('$lines %s /$filters[%d]/om%s',
(lc($opt) eq 'include' ? "=~" : "!~"),
scalar(@filters),
($opt eq lc($opt) ? "i" : "")));
push(@filters, $regexp);
}
# Parse options at compile time so some can become constants to optimize away
BEGIN {
&Getopt::Long::Configure(qw(bundling no_ignore_case));
$getopt_ok = GetOptions("brief|b" => \$brief,
"include|i=s" => \&filter_opt,
"exclude|e=s" => \&filter_opt,
"INCLUDE|I=s" => \&filter_opt,
"EXCLUDE|E=s" => \&filter_opt,
"join|j" => \$join_files,
"no-join|J" => sub { $join_files = 0; },
"sort|s" => \$sort_files,
"loose|l" => sub { $no_loose = ".^"; },
"no-loose|L=s" => \$no_loose,
"trace|t" => \$trace);
}
usage() unless $getopt_ok;
usage("--trace is incompatible with --join.\n") if $trace && $join_files;
$join_files = 1 if $sort_files && !defined($join_files);
use constant BRIEF => !!$brief;
use constant LOOSE => defined($no_loose) && ($no_loose eq ".^" ? 2 : 1);
# Build sub out(header, connection number) to output one connection's data
my $out_body = (LOOSE
? ' if (@loose) { print "\n", @loose; @loose = (); } '
: '');
$out_body .= ' print "\n", $_[0], $lines; ';
$out_body = " if (" . join("\n && ", @conditions) . ") {\n$out_body\n}"
if @conditions;
eval <<EOM;
sub out {
my \$lines = delete(\$conns{\$_[1]});
$out_body
}
1;
EOM
die $@ if $@;
# Read and output log lines from one file
sub do_file {
local(@ARGV) = @_;
my($conn, $line, $act);
while (<>) {
if (BRIEF
? (($conn, $line, $act) = /\bconn=(\d+) (\S+ (\S+).*\n)/)
: (($conn, $act) = /\bconn=(\d+) \S+ (\S+)/ )) {
$conns{$conn} .= (BRIEF ? $line : $_);
out("", $conn) if $act eq 'closed';
} elsif (LOOSE && (LOOSE > 1 || !/$no_loose/omi)) {
s/^\w{3} [ \d]+:\d\d:\d\d [^:]*: // if BRIEF;
push(@loose, $_);
}
}
final() unless $join_files;
}
# Output log lines for unfinished connections
sub final {
if (%conns) {
for my $conn (sort keys %conns) {
out("UNFINISHED:\n", $conn);
}
die if %conns;
}
if (LOOSE && @loose) { print "\n", @loose; @loose = (); }
}
# Main program
if (!@ARGV) {
# Read from stdin
do_file();
} else {
if ($sort_files && @ARGV > 1) {
# Sort files by last modified time; oldest first
my @fileinfo;
for my $file (@ARGV) {
my $age = -M $file;
if (defined($age)) {
push(@fileinfo, [$age, $file]);
} else {
print STDERR "File not found: $file\n";
}
}
exit(1) unless @fileinfo;
@ARGV = map { $_->[1] } sort { $b->[0] <=> $a->[0] } @fileinfo;
}
# Prepare to pipe .gz, .bz2 and .bz files through gunzip or bunzip2
my %type2prog = ("gz" => "gunzip", "bz2" => "bunzip2", "bz" => "bunzip2");
for (@ARGV) {
if (/\.(gz|bz2?)$/) {
my $type = $1;
die "Bad filename: $_\n" if /^[+-]|[^\w\/.,:%=+-]|^$/;
$_ = "$type2prog{$type} -c $_ |";
}
}
# Process the files
for my $file (@ARGV) {
print "\n$file:\n" if $trace;
do_file($file);
}
}
final();
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Please register or to comment