Files
wlan-lanforge-scripts/portal-check.pl

236 lines
6.4 KiB
Perl
Executable File

#!/usr/bin/perl -w
=pod
-------------------------------------------------------------------------------
Use this script to survey a captive portal station:
* check DNS settings via dig
* check initial page redirect with curl
## (C) 2017, Candela Technologies Inc. support@candelatech.com
-------------------------------------------------------------------------------
=cut
package main;
use strict;
use warnings;
use diagnostics;
use Carp;
#$SIG{ __DIE__ } = sub { Carp::confess( @_ ) };
#$SIG{ __WARN__ } = sub { Carp::confess( @_ ) };
use POSIX qw(ceil floor);
use Scalar::Util; #::looks_like_number;
use Getopt::Long;
use Socket;
# Un-buffer output
$| = 1;
use Cwd qw(getcwd);
my $cwd = getcwd();
# this is pedantic necessity for the following use statements
if ( $cwd =~ q(.*LANforge-Server\scripts$)) {
use lib '/home/lanforge/scripts'
}
else {
use lib '/home/lanforge/scripts';
}
use List::Util qw(first);
use LANforge::Endpoint;
use LANforge::Port;
use LANforge::Utils;
use Net::Telnet ();
my $usage = qq($0 [--mgr {host-name | IP}]
[--mgr_port {ip port}] # use if on non-default management port
[--resource {resource}] # use if multiple lanforge systems; defaults to 1
[--quiet { yes | no }] # debug output; -q
[--log_cli] # enables CLI command printing to STDOUT
# same effect when setting env var LOG_CLI=STDOUT
[--radio {name}] # radio parent of sta1000 e.g. wiphy0
[--sta {name}] # station to use e.g. sta1000 or wlan0
[--ssid {ssid}] # ssid to set station on
[--upstream {dev}] # e.g. eth1 # attempt to ping upstream port from station
[--verbose|v]
);
if (@ARGV < 2) {
print $usage;
exit 0;
}
my $help;
our $lfmgr_host = "localhost";
my $lfmgr_port = 4001;
our $resource = 1;
our $quiet = "yes";
our $sta_wiphy = "wiphy0";
our $ssid = "";
our $sta;
our $upstream_port = "";
my $log_cli;
our $verbose = 0;
GetOptions
(
'mgr|m=s' => \$::lfmgr_host,
'mgr_port|p=i' => \$lfmgr_port,
'resource|r=i' => \$::resource,
'quiet|q=s' => \$::quiet,
'radio|o=s' => \$::sta_wiphy,
'ssid|s=s' => \$::ssid,
'upstream|t=s' => \$::upstream_port,
'log_cli=s{0,1}' => \$log_cli, # use ENV{LOG_CLI} elsewhere
'sta|w=s' => \$::sta,
'verbose|v=i' => \$::verbose,
'help|?' => \$help,
) || (print($usage) && exit(1));
if ($help) {
print($usage) && exit(0);
}
$SIG{ __DIE__ } = sub { Carp::confess( @_ ) };
if ($::quiet eq "0") {
$::quiet = "no";
}
elsif ($::quiet eq "1") {
$::quiet = "yes";
}
# Open connection to the LANforge server.
if (defined $log_cli) {
if ($log_cli ne "unset") {
# here is how we reset the variable if it was used as a flag
if ($log_cli eq "") {
$ENV{'LOG_CLI'} = 1;
}
else {
$ENV{'LOG_CLI'} = $log_cli;
}
}
}
our $t = new Net::Telnet(Prompt => '/default\@btbits\>\>/',
Timeout => 20);
$t->open(Host => $lfmgr_host,
Port => $lfmgr_port,
Timeout => 10);
$t->waitfor("/btbits\>\>/");
# Configure our utils.
our $utils = new LANforge::Utils();
$utils->telnet($t); # Set our telnet object.
if ($utils->isQuiet()) {
if (defined $ENV{'LOG_CLI'} && $ENV{'LOG_CLI'} ne "") {
$utils->cli_send_silent(0);
}
else {
$utils->cli_send_silent(1); # Do not show input to telnet
}
$utils->cli_rcv_silent(1); # Repress output from telnet
}
else {
$utils->cli_send_silent(0); # Show input to telnet
$utils->cli_rcv_silent(0); # Show output from telnet
}
$utils->log_cli("# $0 ".`date "+%Y-%m-%d %H:%M:%S"`);
# this is the --show_port options ("")
my @port_txt = ();
our $port_mac = '';
our $port_dev = '';
our $port_ip = '';
our $port_dns = '';
our $port_gateway = '';
if ((defined $::sta) && ("$sta" ne "")) {
@port_txt = split("\n", $utils->doAsyncCmd("nc_show_port 1 $::resource $::sta"));
}
my @likely = ();
if ($::verbose > 0) {
print "--------------------------------------------\n";
print join("\n", @port_txt);
print "--------------------------------------------\n";
}
@likely = grep {/MAC: .* DEV: .*/} @port_txt;
die("Unable to find port named $::sta")
if (!@likely);
if (@likely) {
($port_mac) = $likely[0] =~ /MAC: ([^ ]+) /;
$port_mac = ''
if (!defined $port_ip);
($port_dev) = $likely[0] =~ /DEV: ([^ ]+) /;
$port_dev = ''
if (!defined $port_ip);
}
@likely = grep {/IP: .* MASK:/} @port_txt;
if (@likely) {
($port_ip) = $likely[0] =~ /IP: ([^ ]+) /;
$port_ip = ''
if (!defined $port_ip);
}
@likely = grep {/DNS Servers:/} @port_txt;
if (@likely) {
if ($verbose) {
print "LIKELY DNS SERVERS:\n";
print join("\n", @likely)."\n";
}
($port_dns) = $likely[0] =~ /DNS Servers: (.*)$/;
$port_dns =''
if (!defined $port_dns)
}
@likely = grep {/GW: /} @port_txt;
if (@likely) {
($port_gateway) = $likely[0] =~ /GW: ([^ ]+) /;
$port_gateway = ''
if (!defined $port_gateway);
}
print qq(Port name: $sta IP: $port_ip; DNS: $port_dns; GW: $port_gateway;\n\n);
if ($port_ip eq '') {
print "\nNo tests possible, bye\n";
exit(1);
}
if ($port_gateway ne '') {
print "\nChecking gateway: $port_gateway\n";
print("ping -c2 -w2 -t2 -I $port_ip $port_gateway \n");
system("ping -c2 -w2 -t2 -I $port_ip $port_gateway");
}
else {
print "Not checking gateway\n";
}
if ($port_dns ne '') {
my @dns = split(',', $port_dns);
if (@dns) {
print "\nChecking DNS: $dns[0]\n";
print("\nping -c2 -w2 -t2 -I $port_ip $dns[0]\n");
system("ping -c2 -w2 -t2 -I $port_ip $dns[0]");
print("\ndig -b $port_ip -q www.slashdot.org \@$dns[0]\n");
system("dig -b $port_ip -q www.slashdot.org \@$dns[0]");
}
}
else {
print "Not checking DNS\n";
}
sub assembleCurl {
return qq(/home/lanforge/local/bin/curl -svLki -4 -m30 --connect-timeout 15 )
.qq( --interface $port_dev )
.qq( --localaddr $port_ip )
.qq( --dns-interface $port_dev )
.qq( --dns-ipv4-addr $port_ip )
.qq( --dns-servers $::port_dns )
.join(" ", @_);
}
print "\nChecking redirect to http://1.1.1.1/ ...\n";
my $cmd = assembleCurl(qw("http://1.1.1.1/"));
print("\n$cmd\n");
system("$cmd");
print "\n\nChecking redirect to http://www.slashdot.org/ ...\n";
$cmd = assembleCurl(qw("http://www.slashdot.org/"));
print("\n$cmd\n");
system("$cmd");
print "\n\n";
#