2017-02-17 10:51:43 +13:00
#! /usr/bin/perl
#
# Summarise tshark pdml output into a form suitable for the load test tool
#
# Copyright (C) Catalyst.Net Ltd 2017
#
# Catalyst.Net's contribution was written by Gary Lockyer
# <gary@catalyst.net.nz>.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
use warnings ;
use strict ;
use Getopt::Long ;
use Pod::Usage ;
BEGIN {
unless ( eval "require XML::Twig" ) {
warn "traffic_summary requires the perl module XML::Twig\n" .
"on Ubuntu/Debian releases run\n" .
" sudo apt install libxml-twig-perl \n" .
"or install from CPAN\n" .
"\nThe reported error was:\n$@" ;
exit ( 1 ) ;
}
}
my % ip_map ; # Map of IP address to sequence number
my $ ip_sequence = 0 ; # count of unique IP addresses seen
my $ timestamp ; # Packet timestamp
my $ stream ; # Wireshark stream number
2023-06-30 11:57:55 +02:00
my $ ip_proto ; # IP protocol (IANA protocol number)
2017-02-17 10:51:43 +13:00
my $ source ; # source IP address
my $ dest ; # destination address
my $ proto ; # application protocol name
my $ description ; # protocol specific description
my % proto_data ; # protocol specific data captured for the current packet
my $ malformed_packet ; # Indicates the current packet has errors
my $ ldap_filter ; # cleaned ldap filter
my $ ldap_attributes ; # attributes requested in an ldap query
# Dispatch table mapping the wireshark variables of interest to the
# functions responsible for processing them
my % field_dispatch_table = (
'timestamp' = > \ & timestamp ,
'ip.src' = > \ & ip_src ,
'ipv6.src' = > \ & ip_src ,
'ip.dst' = > \ & ip_dst ,
'ipv6.dst' = > \ & ip_dst ,
'ip.proto' = > \ & ip_proto ,
'udp.stream' = > \ & stream ,
'tcp.stream' = > \ & stream ,
'dns.flags.opcode' = > \ & field_data ,
'dns.flags.response' = > \ & field_data ,
'netlogon.opnum' = > \ & field_data ,
'kerberos.msg_type' = > \ & field_data ,
'smb.cmd' = > \ & field_data ,
'smb2.cmd' = > \ & field_data ,
'ldap.protocolOp' = > \ & field_data ,
'gss-api.OID' = > \ & field_data ,
'ldap.gssapi_encrypted_payload' = > \ & field_data ,
'ldap.baseObject' = > \ & field_data ,
'ldap.scope' = > \ & field_data ,
'ldap.AttributeDescription' = > \ & ldap_attribute ,
'ldap.modification_element' = > \ & ldap_add_modify ,
'ldap.AttributeList_item_element' = > \ & ldap_add_modify ,
'ldap.operation' = > \ & field_data ,
'ldap.authentication' = > \ & field_data ,
'lsarpc.opnum' = > \ & field_data ,
'samr.opnum' = > \ & field_data ,
'dcerpc.pkt_type' = > \ & field_data ,
'epm.opnum' = > \ & field_data ,
'dnsserver.opnum' = > \ & field_data ,
'drsuapi.opnum' = > \ & field_data ,
'browser.command' = > \ & field_data ,
'smb_netlogon.command' = > \ & field_data ,
'srvsvc.opnum' = > \ & field_data ,
'nbns.flags.opcode' = > \ & field_data ,
'nbns.flags.response' = > \ & field_data ,
'_ws.expert.message' = > \ & field_data ,
) ;
# Dispatch table mapping protocols to the routine responsible for formatting
# their output. Protocols not in this table are ignored.
#
my % proto_dispatch_table = (
'dns' = > sub { return format_opcode ( 'dns.flags.response' ) } ,
'rpc_netlogon' = > sub { return format_opcode ( 'netlogon.opnum' ) } ,
'kerberos' = > \ & format_kerberos ,
'smb' = > sub { return format_opcode ( 'smb.cmd' ) } ,
'smb2' = > sub { return format_opcode ( 'smb2.cmd' ) } ,
'ldap' = > \ & format_ldap ,
'cldap' = > \ & format_ldap ,
'lsarpc' = > sub { return format_opcode ( 'lsarpc.opnum' ) } ,
'samr' = > sub { return format_opcode ( 'samr.opnum' ) } ,
'dcerpc' = > sub { return format_opcode ( 'dcerpc.pkt_type' ) } ,
'epm' = > sub { return format_opcode ( 'epm.opnum' ) } ,
'dnsserver' = > sub { return format_opcode ( 'dnsserver.opnum' ) } ,
'drsuapi' = > sub { return format_opcode ( 'drsuapi.opnum' ) } ,
'browser' = > sub { return format_opcode ( 'browser.command' ) } ,
'smb_netlogon' = > sub { return format_opcode ( 'smb_netlogon.command' ) } ,
'srvsvc' = > sub { return format_opcode ( 'srvsvc.opnum' ) } ,
'nbns' = > sub { return format_opcode ( 'nbns.flags.response' ) } ,
) ;
# XPath entry to extract the kerberos cname
my $ kerberos_cname_path =
'packet/proto/field[@name = "kerberos.as_req_element"]'
. '/field[@name = "kerberos.req_body_element"]'
. '/field[@name = "kerberos.cname_element"]'
. '/field[@name = "kerberos.name_string"]'
. '/field[@name = "kerberos.KerberosString"]' ;
# XPath entry to extract the ldap filter
my $ ldap_filter_path =
'field[@name = "ldap.searchRequest_element"]/field' ;
# Create an XML Twig parser and register the event handlers.
#
my $ t = XML::Twig - > new (
start_tag_handlers = > {
'packet' = > \ & packet_start ,
} ,
twig_handlers = > {
'packet' = > \ & packet ,
'proto' = > \ & protocol ,
'field' = > \ & field ,
$ kerberos_cname_path = > \ & kerberos_cname ,
$ ldap_filter_path = > \ & ldap_filter ,
} ,
) ;
#------------------------------------------------------------------------------
# Main loop
#
#------------------------------------------------------------------------------
my $ help = 0 ;
GetOptions ( 'help|h' = > \ $ help ) or pod2usage ( 2 ) ;
pod2usage ( 1 ) if $ help ;
if ( @ ARGV ) {
foreach my $ file ( @ ARGV ) {
eval {
$ t - > parsefile ( $ file ) ;
} ;
if ( $@ ) {
print STDERR "Unable to process $file, " .
"did you run tshark with the -T pdml option?" ;
}
}
} else {
pod2usage ( 1 ) if - t STDIN ;
eval {
$ t - > parse ( \ * STDIN ) ;
} ;
if ( $@ ) {
print STDERR "Unable to process input, " .
"are you running tshark with the -T pdml option?" ;
}
}
#------------------------------------------------------------------------------
# New packet detected reset the globals
#------------------------------------------------------------------------------
sub packet_start
{
my ( $ t , $ packet ) = @ _ ;
$ timestamp = "" ;
$ stream = "" ;
$ ip_proto = "" ;
$ source = "" ;
$ dest = "" ;
$ description = undef ;
% proto_data = ( ) ;
$ malformed_packet = undef ;
$ ldap_filter = "" ;
$ ldap_attributes = "" ;
}
#------------------------------------------------------------------------------
# Complete packet element parsed from the XML feed
# output the protocol summary if required
#------------------------------------------------------------------------------
sub packet
{
my ( $ t , $ packet ) = @ _ ;
my $ data ;
if ( exists $ proto_dispatch_table { $ proto } ) {
if ( $ malformed_packet ) {
$ data = "\t\t** Malformed Packet ** " . ( $ proto_data { '_ws.expert.message.show' } || '' ) ;
} else {
my $ rsub = $ proto_dispatch_table { $ proto } ;
$ data = & $ rsub ( ) ;
}
print "$timestamp\t$ip_proto\t$stream\t$source\t$dest\t$proto\t$data\n" ;
}
$ t - > purge ;
}
#------------------------------------------------------------------------------
# Complete protocol element parsed from the XML input
# Update the protocol name
#------------------------------------------------------------------------------
sub protocol
{
my ( $ t , $ protocol ) = @ _ ;
if ( $ protocol - > { att } - > { showname } ) {
}
# Tag a packet as malformed if the protocol is _ws.malformed
# and the hide attribute is not 'yes'
if ( $ protocol - > { att } - > { name } eq '_ws.malformed'
&& ! ( $ protocol - > { att } - > { hide } && $ protocol - > { att } - > { hide } eq 'yes' )
) {
$ malformed_packet = 1 ;
}
# Don't set the protocol name if it's a wireshark malformed
# protocol entry, or the packet was truncated during capture
my $ p = $ protocol - > { att } - > { name } ;
if ( $ p ne '_ws.malformed' && $ p ne '_ws.short' ) {
$ proto = $ p ;
}
}
#------------------------------------------------------------------------------
# Complete field element parsed, extract any data of interest
#------------------------------------------------------------------------------
sub field
{
my ( $ t , $ field ) = @ _ ;
my $ name = $ field - > { att } - > { name } ;
# Only process the field if it has a corresponding entry in
# %field_dispatch_table
if ( exists $ field_dispatch_table { $ name } ) {
my $ rsub = $ field_dispatch_table { $ name } ;
& $ rsub ( $ field ) ;
}
}
#------------------------------------------------------------------------------
# Process a timestamp field element
#------------------------------------------------------------------------------
sub timestamp
{
my ( $ field ) = @ _ ;
$ timestamp = $ field - > { att } - > { value } ;
}
#------------------------------------------------------------------------------
# Process a wireshark stream element, used to group a sequence of requests
# and responses between two IP addresses
#------------------------------------------------------------------------------
sub stream
{
my ( $ field ) = @ _ ;
$ stream = $ field - > { att } - > { show } ;
}
#------------------------------------------------------------------------------
# Process a source ip address field, mapping the IP address to it's
# corresponding sequence number.
#------------------------------------------------------------------------------
sub ip_src
{
my ( $ field ) = @ _ ;
$ source = map_ip ( $ field ) ;
}
#------------------------------------------------------------------------------
# Process a destination ip address field, mapping the IP address to it's
# corresponding sequence number.
#------------------------------------------------------------------------------
sub ip_dst
{
my ( $ field ) = @ _ ;
$ dest = map_ip ( $ field ) ;
}
#------------------------------------------------------------------------------
# Process an ip protocol element, extracting IANA protocol number
#------------------------------------------------------------------------------
sub ip_proto
{
my ( $ field ) = @ _ ;
$ ip_proto = $ field - > { att } - > { value } ;
}
#------------------------------------------------------------------------------
# Extract an ldap attribute and append it to ldap_attributes
#------------------------------------------------------------------------------
sub ldap_attribute
{
my ( $ field ) = @ _ ;
my $ attribute = $ field - > { att } - > { show } ;
if ( defined $ attribute ) {
$ ldap_attributes . = "," if $ ldap_attributes ;
$ ldap_attributes . = $ attribute ;
}
}
#------------------------------------------------------------------------------
# Process a field element, extract the value, show and showname attributes
# and store them in the %proto_data hash.
#
#------------------------------------------------------------------------------
sub field_data
{
my ( $ field ) = @ _ ;
my $ name = $ field - > { att } - > { name } ;
$ proto_data { $ name . '.value' } = $ field - > { att } - > { value } ;
$ proto_data { $ name . '.show' } = $ field - > { att } - > { show } ;
$ proto_data { $ name . '.showname' } = $ field - > { att } - > { showname } ;
}
#------------------------------------------------------------------------------
# Process a kerberos cname element, if the cname ends with a $ it's a machine
# name. Otherwise it's a user name.
#
#------------------------------------------------------------------------------
sub kerberos_cname
{
my ( $ t , $ field ) = @ _ ;
my $ cname = $ field - > { att } - > { show } ;
my $ type ;
if ( $ cname =~ /\$$/ ) {
$ type = 'machine' ;
} else {
$ type = 'user' ;
}
$ proto_data { 'kerberos.cname.type' } = $ type ;
}
#------------------------------------------------------------------------------
# Process an ldap filter, remove the values but keep the attribute names
#------------------------------------------------------------------------------
sub ldap_filter
{
my ( $ t , $ field ) = @ _ ;
if ( $ field - > { att } - > { show } && $ field - > { att } - > { show } =~ /^Filter:/ ) {
my $ filter = $ field - > { att } - > { show } ;
# extract and save the objectClass to keep the value
my @ object_classes ;
while ( $ filter =~ m/\((objectClass=.*?)\)/g ) {
push @ object_classes , $ 1 ;
}
# extract and save objectCategory and the top level value
my @ object_categories ;
while ( $ filter =~ m/(\(objectCategory=.*?,|\(objectCategory=.*?\))/g
) {
push @ object_categories , $ 1 ;
}
# Remove all the values from the attributes
# Input
# Filter: (nCName=DC=DomainDnsZones,DC=sub1,DC=ad,DC=rh,DC=at,DC=net)
# Output
# (nCName)
$ filter =~ s/^Filter:\s*// ; # Remove the 'Filter: ' prefix
$ filter =~ s/=.*?\)/\)/g ; # Remove from the = to the first )
# Now restore the parts of objectClass and objectCategory that are being
# retained
#
for my $ cat ( @ object_categories ) {
$ filter =~ s/\(objectCategory\)/$cat/ ;
}
for my $ class ( @ object_classes ) {
$ filter =~ s/\(objectClass\)/($class)/ ;
}
$ ldap_filter = $ filter ;
} else {
# Ok not an ldap filter so call the default field handler
field ( $ t , $ field ) ;
}
}
#------------------------------------------------------------------------------
# Extract the attributes from ldap modification and add requests
#------------------------------------------------------------------------------
sub ldap_add_modify
{
my ( $ field ) = @ _ ;
my $ type = $ field - > first_child ( 'field[@name="ldap.type"]' ) ;
my $ attribute = $ type - > { att } - > { show } if $ type ;
if ( defined $ attribute ) {
$ ldap_attributes . = "," if $ ldap_attributes ;
$ ldap_attributes . = $ attribute ;
}
}
#------------------------------------------------------------------------------
# Map an IP address to a unique sequence number. Assigning it a sequence number
# if one has not already been assigned.
#
#------------------------------------------------------------------------------
sub map_ip
{
my ( $ field ) = @ _ ;
my $ ip = $ field - > { att } - > { show } ;
if ( ! exists ( $ ip_map { $ ip } ) ) {
$ ip_sequence + + ;
$ ip_map { $ ip } = $ ip_sequence ;
}
return $ ip_map { $ ip } ;
}
#------------------------------------------------------------------------------
# Format a protocol operation code for output.
#
#------------------------------------------------------------------------------
sub format_opcode
{
my ( $ name ) = @ _ ;
my $ operation = $ proto_data { $ name . '.show' } ;
my $ description = $ proto_data { $ name . '.showname' } || '' ;
# Strip off the common prefix text, and the trailing (n).
# This tidies up most but not all descriptions.
$ description =~ s/^[^:]*?: ?// if $ description ;
$ description =~ s/^Message is a // if $ description ;
$ description =~ s/\(\d+\)\s*$// if $ description ;
$ description =~ s/\s*$// if $ description ;
return "$operation\t$description" ;
}
#------------------------------------------------------------------------------
# Format ldap protocol details for output
#------------------------------------------------------------------------------
sub format_ldap
{
my ( $ name ) = @ _ ;
if ( exists ( $ proto_data { 'ldap.protocolOp.show' } )
|| exists ( $ proto_data { 'gss-api.OID.show' } )
) {
my $ operation = $ proto_data { 'ldap.protocolOp.show' } ;
my $ description = $ proto_data { 'ldap.protocolOp.showname' } || '' ;
my $ oid = $ proto_data { 'gss-api.OID.show' } || '' ;
my $ base_object = $ proto_data { 'ldap.baseObject.show' } || '' ;
my $ scope = $ proto_data { 'ldap.scope.show' } || '' ;
# Now extract operation specific data
my $ extra ;
my $ extra_desc ;
$ operation = '' if ! defined $ operation ;
if ( $ operation eq 6 ) {
# Modify operation
$ extra = $ proto_data { 'ldap.operation.show' } ;
$ extra_desc = $ proto_data { 'ldap.operation.showname' } ;
} elsif ( $ operation eq 0 ) {
# Bind operation
$ extra = $ proto_data { 'ldap.authentication.show' } ;
$ extra_desc = $ proto_data { 'ldap.authentication.showname' } ;
}
$ extra = '' if ! defined $ extra ;
$ extra_desc = '' if ! defined $ extra_desc ;
# strip the values out of the base object
if ( $ base_object ) {
$ base_object =~ s/^<// ; # leading '<' if present
$ base_object =~ s/>$// ; # trailing '>' if present
$ base_object =~ s/=.*?,/,/g ; # from = up to the next comma
$ base_object =~ s/=.*?$// ; # from = up to the end of string
}
# strip off the leading prefix on the extra_description
# and the trailing (n);
$ extra_desc =~ s/^[^:]*?: ?// if $ extra_desc ;
$ extra_desc =~ s/\(\d+\)\s*$// if $ extra_desc ;
$ extra_desc =~ s/\s*$// if $ extra_desc ;
# strip off the common prefix on the description
# and the trailing (n);
$ description =~ s/^[^:]*?: ?// if $ description ;
$ description =~ s/\(\d+\)\s*$// if $ description ;
$ description =~ s/\s*$// if $ description ;
return "$operation\t$description\t$scope\t$base_object"
. "\t$ldap_filter\t$ldap_attributes\t$extra\t$extra_desc\t$oid" ;
} else {
return "\t*** Unknown ***" ;
}
}
#------------------------------------------------------------------------------
# Format kerberos protocol details for output.
#------------------------------------------------------------------------------
sub format_kerberos
{
2017-06-23 14:16:53 +12:00
my $ msg_type = $ proto_data { 'kerberos.msg_type.show' } || '' ;
2017-02-17 10:51:43 +13:00
my $ cname_type = $ proto_data { 'kerberos.cname.type' } || '' ;
my $ description = $ proto_data { 'kerberos.msg_type.showname' } || '' ;
# Tidy up the description
$ description =~ s/^[^:]*?: ?// if $ description ;
$ description =~ s/\(\d+\)\s*$// if $ description ;
$ description =~ s/\s*$// if $ description ;
return "$msg_type\t$description\t$cname_type" ;
}
= pod
= head1 NAME
traffic_summary . pl - summarise tshark pdml output
= head1 USAGE
B <traffic_summary.pl> [ FILE ... ]
Summarise samba network traffic from tshark pdml output . Produces a tsv
delimited summary of samba activity .
To process unencrypted traffic
tshark - r capture . file - T pdml | traffic_summary . pl
To process encrypted kerberos traffic
tshark - r capture . file - K krb5 . keytab - o kerberos . decrypt:true - T pdml | traffic_summary . pl
To display more detailed documentation , including details of the output format
2017-03-01 17:33:09 +13:00
perldoc traffic_summary . pl
2017-02-17 10:51:43 +13:00
NOTE: tshark pdml output is very verbose , so it ' s better to pipe the tshark
output directly to traffic_summary , rather than generating
intermediate pdml format files .
= head1 OPTIONS
B <--help> Display usage message and exit .
= head1 DESCRIPTION
Summarises tshark pdml output into a format suitable for load analysis
and input into load generation tools .
It reads the pdml input from stdin or the list of files passed on the command line .
= head2 Output format
The output is tab delimited fields and one line per summarised packet .
= head3 Fields
B <timestamp> Packet timestamp
B < IP protocol > The IANA protocol number
B < Wireshark Stream Number > Calculated by wireshark groups related requests and responses
B < Source IP > The unique sequence number for the source IP address
B < Destination IP > The unique sequence number for the destination IP address
2023-06-30 11:57:55 +02:00
B <protocol> The protocol name
2017-02-17 10:51:43 +13:00
B <opcode> The protocol operation code
B <Description> The protocol or operation description
B <extra> Extra protocol specific data , may be more than one field
= head2 IP address mapping
Rather than capturing and printing the IP addresses . Each unique IP address
seen is assigned a sequence number . So the first IP address seen will be 1 ,
the second 2 ...
= head2 Packets collected
Packets containing the following protocol records are summarised:
dns
rpc_netlogon
kerberos
smb
smb2
ldap
cldap
lsarpc
samr
dcerpc
epm
dnsserver
drsuapi
browser
smb_netlogon
srvsvc
nbns
Any other packets are ignored .
In addition to the standard elements extra data is returned for the following
protocol record .
= head3 kerberos
cname_type machine cname ends with a $
user cname does not end with a $
= head3 ldap
scope Query Scope
0 - Base
1 - One level
2 - sub tree
base_object ldap base object
ldap_filter the ldap filter , attribute names are retained but the values
are removed .
ldap_attributes ldap attributes , only the names are retained any values are
discarded , with the following two exceptions
objectClass all the attribute values are retained
objectCategory the top level value is retained
i . e . everything from the = to the first ,
= head3 ldap modifiyRequest
In addition to the standard ldap fields the modification type is also captured
2023-06-30 11:57:55 +02:00
modify_operator for modifyRequests this contains the modify operation
2017-02-17 10:51:43 +13:00
0 - add
1 - delete
2 - replace
modify_description a description of the operation if available
= head3 modify bindRequest
In addition to the standard ldap fields details of the authentication
type are captured
authentication type 0 - Simple
3 - SASL
description Description of the authentication mechanism
oid GSS - API OID ' s
1.2 .840 .113554 .1 .2 .2 - Kerberos v5
1.2 .840 .48018 .1 .2 .2 - Kerberos V5
( incorrect , used by old Windows versions )
1.3 .6 .1 .5 .5 .2 - SPNEGO
1.3 .6 .1 .5 .2 .5 - IAKERB
1.3 .6 .1 .4 .1 .311 .2 .2 .10 - NTLM SSP
1.3 .6 .1 .5 .5 .14 - SCRAM - SHA - 1
1.3 .6 .1 .5 .5 .18 - SCRAM - SHA - 256
1.3 .6 .1 .5 .5 .15 .1 .1 . * - GSS - EAP
1.3 .6 .1 .5 .2 .7 - PKU2U
1.3 .6 .1 .5 .5 .1 .1 - SPKM - 1
1.3 .6 .1 .5 .5 .1 .2 - SPKM - 2
1.3 .6 .1 .5 .5 .1 .3 - SPKM - 3
1.3 .6 .1 .5 .5 .9 - LIPKEY
1.2 .752 .43 .14 .2 - NETLOGON
= head1 DEPENDENCIES
tshark
XML:: Twig For Ubuntu libxml - twig - perl , or from CPAN
use Getopt::Long
use Pod::Usage
= head1 Diagnostics
= head2 ** Unknown **
Unable to determine the operation being performed , for ldap it typically
indicates a kerberos encrypted operation .
= head2 ** Malformed Packet **
tshark indicated that the packet was malformed , for ldap it usually indicates TLS
encrypted traffic .
2023-06-30 11:57:55 +02:00
= head1 LICENSE AND COPYRIGHT
2017-02-17 10:51:43 +13:00
Copyright ( C ) Catalyst . Net Ltd 2017
Catalyst . Net ' s contribution was written by Gary Lockyer
<gary@catalyst.net.nz> .
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see <http://www.gnu.org/licenses/> .
= cut