Skip to content
Snippets Groups Projects

Compare revisions

Changes are shown as if the source revision was being merged into the target revision. Learn more about comparing revisions.

Source

Select target project
No results found

Target

Select target project
  • opentrafficshaper/opentrafficshaper
  • eezysol/opentrafficshaper
  • sudesh/opentrafficshaper
  • Yuriy/opentrafficshaper
  • nkukard/opentrafficshaper
  • fcardoso/opentrafficshaper
6 results
Show changes
Showing
with 4214 additions and 2926 deletions
...@@ -5,7 +5,7 @@ DROP TABLE IF EXISTS identifiers; ...@@ -5,7 +5,7 @@ DROP TABLE IF EXISTS identifiers;
CREATE TABLE identifiers ( CREATE TABLE identifiers (
`ID` SERIAL, `ID` SERIAL,
`Identifier` VARCHAR(255) NOT NULL `Identifier` VARCHAR(255) NOT NULL
) Engine=MyISAM; ) Engine=InnoDB;
/* For queries */ /* For queries */
CREATE INDEX identifiers_idx1 ON identifiers (`Identifier`); CREATE INDEX identifiers_idx1 ON identifiers (`Identifier`);
...@@ -28,12 +28,12 @@ CREATE TABLE stats ( ...@@ -28,12 +28,12 @@ CREATE TABLE stats (
`Limit` MEDIUMINT UNSIGNED NOT NULL, `Limit` MEDIUMINT UNSIGNED NOT NULL,
`Rate` MEDIUMINT UNSIGNED NOT NULL, `Rate` MEDIUMINT UNSIGNED NOT NULL,
`PPS` MEDIUMINT UNSIGNED NOT NULL, `PPS` MEDIUMINT UNSIGNED NOT NULL,
`Queue_Len` MEDIUMINT UNSIGNED NOT NULL, `QueueLen` MEDIUMINT UNSIGNED NOT NULL,
`Total_Bytes` BIGINT UNSIGNED NOT NULL, `TotalBytes` BIGINT UNSIGNED NOT NULL,
`Total_Packets` BIGINT UNSIGNED NOT NULL, `TotalPackets` BIGINT UNSIGNED NOT NULL,
`Total_Overlimits` BIGINT UNSIGNED NOT NULL, `TotalOverlimits` BIGINT UNSIGNED NOT NULL,
`Total_Dropped` BIGINT UNSIGNED NOT NULL `TotalDropped` BIGINT UNSIGNED NOT NULL
) Engine=MyISAM; ) Engine=InnoDB;
/* For queries */ /* For queries */
CREATE INDEX stats_idx1 ON stats (`IdentifierID`); CREATE INDEX stats_idx1 ON stats (`IdentifierID`);
...@@ -56,7 +56,7 @@ CREATE TABLE stats_basic ( ...@@ -56,7 +56,7 @@ CREATE TABLE stats_basic (
`Timestamp` INTEGER UNSIGNED NOT NULL, `Timestamp` INTEGER UNSIGNED NOT NULL,
`Counter` BIGINT UNSIGNED NOT NULL `Counter` BIGINT UNSIGNED NOT NULL
) Engine=MyISAM; ) Engine=InnoDB;
/* For queries */ /* For queries */
CREATE INDEX stats_basic_idx1 ON stats (`IdentifierID`); CREATE INDEX stats_basic_idx1 ON stats (`IdentifierID`);
......
# Basic radius dictionary # Basic radius dictionary
# Copyright (C) 2009-2013, AllWorldIT # Copyright (C) 2009-2015, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
......
# AllWorldIT vendor radius dictionary # AllWorldIT vendor radius dictionary
# Copyright (C) 2009-2013, AllWorldIT # Copyright (C) 2009-2015, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -21,4 +21,5 @@ VENDOR AllWorldIT 42109 ...@@ -21,4 +21,5 @@ VENDOR AllWorldIT 42109
ATTRIBUTE OpenTrafficShaper-Traffic-Limit 1 string AllWorldIT ATTRIBUTE OpenTrafficShaper-Traffic-Limit 1 string AllWorldIT
ATTRIBUTE OpenTrafficShaper-Traffic-Group 2 integer AllWorldIT ATTRIBUTE OpenTrafficShaper-Traffic-Group 2 integer AllWorldIT
ATTRIBUTE OpenTrafficShaper-Traffic-Class 3 integer AllWorldIT ATTRIBUTE OpenTrafficShaper-Traffic-Class 3 integer AllWorldIT
ATTRIBUTE OpenTrafficShaper-Traffic-Pool 4 string AllWorldIT
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
# PID file to write our PID to # PID file to write our PID to
# #
# default: # default:
# pid_file=/var/run/opentrafficshaper/opentrafficshaper.pid # pid_file=/run/opentrafficshaper/opentrafficshaper.pid
# State file, this file is used to store persistent information # State file, this file is used to store persistent information
......
# POE::Filter::HybridHTTP - Copyright 2013, AllworldIT # POE::Filter::HybridHTTP - Copyright 2007-2023, AllworldIT
# Hybrid HTTP filter supporting websockets too. # Hybrid HTTP filter supporting websockets too.
#
# 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/>.
## ##
# Code originally based on POE::Filter::HTTPD # Code originally based on POE::Filter::HTTPD
...@@ -15,7 +28,7 @@ ...@@ -15,7 +28,7 @@
# and from HTTPD filters, they should submit their request as a patch. # and from HTTPD filters, they should submit their request as a patch.
## ##
package POE::Filter::HybridHTTP; package opentrafficshaper::POE::Filter::HybridHTTP;
use warnings; use warnings;
use strict; use strict;
...@@ -23,11 +36,11 @@ use strict; ...@@ -23,11 +36,11 @@ use strict;
use bytes; use bytes;
use POE::Filter; use POE::Filter;
use POE::Filter::HybridHTTP::WebSocketFrame; use opentrafficshaper::POE::Filter::HybridHTTP::WebSocketFrame;
use vars qw($VERSION @ISA); use vars qw($VERSION @ISA);
# NOTE - Should be #.### (three decimal places) # NOTE - Should be #.### (three decimal places)
$VERSION = '1.000'; $VERSION = '2.000';
@ISA = qw(POE::Filter); @ISA = qw(POE::Filter);
...@@ -52,11 +65,11 @@ my $HTTP_1_1 = _http_version("HTTP/1.1"); ...@@ -52,11 +65,11 @@ my $HTTP_1_1 = _http_version("HTTP/1.1");
# Class instantiation # Class instantiation
sub new sub new
{ {
my $class = shift; my $class = shift;
# These are our internal properties # These are our internal properties
my $self = { }; my $self = { };
# Build our class # Build our class
bless($self, $class); bless($self, $class);
...@@ -71,7 +84,7 @@ sub new ...@@ -71,7 +84,7 @@ sub new
# From the docs: # From the docs:
# get_one_start() accepts an array reference containing unprocessed stream chunks. The chunks are added to the filter's Internal # get_one_start() accepts an array reference containing unprocessed stream chunks. The chunks are added to the filter's Internal
# buffer for parsing by get_one(). # buffer for parsing by get_one().
sub get_one_start sub get_one_start
{ {
my ($self, $stream) = @_; my ($self, $stream) = @_;
...@@ -84,7 +97,7 @@ sub get_one_start ...@@ -84,7 +97,7 @@ sub get_one_start
# This is called to see if we can grab records/items # This is called to see if we can grab records/items
sub get_one sub get_one
{ {
my $self = shift; my $self = shift;
...@@ -96,7 +109,7 @@ sub get_one ...@@ -96,7 +109,7 @@ sub get_one
# Waiting for content. # Waiting for content.
} elsif ($self->{'state'} == ST_HTTP_CONTENT) { } elsif ($self->{'state'} == ST_HTTP_CONTENT) {
return $self->_get_one_http_content(); return $self->_get_one_http_content();
# Websocket # Websocket
} elsif ($self->{'state'} == ST_WEBSOCKET_STREAM) { } elsif ($self->{'state'} == ST_WEBSOCKET_STREAM) {
return $self->_get_one_websocket_record(); return $self->_get_one_websocket_record();
...@@ -109,7 +122,7 @@ sub get_one ...@@ -109,7 +122,7 @@ sub get_one
# Function to push data to the socket # Function to push data to the socket
sub put sub put
{ {
my ($self, $responses) = @_; my ($self, $responses) = @_;
my @results; my @results;
...@@ -124,7 +137,7 @@ sub put ...@@ -124,7 +137,7 @@ sub put
# Check if its a websocket upgrade # Check if its a websocket upgrade
if ( if (
# Is it a request and do we have a original request? # Is it a request and do we have a original request?
$h_upgrade eq "websocket" && defined($self->{'last_request'}) && $h_upgrade eq "websocket" && defined($self->{'last_request'}) &&
# If so was there a websocket-key? # If so was there a websocket-key?
(my $websocketKey = $self->{'last_request'}->header('Sec-WebSocket-Key')) (my $websocketKey = $self->{'last_request'}->header('Sec-WebSocket-Key'))
) { ) {
...@@ -132,7 +145,7 @@ sub put ...@@ -132,7 +145,7 @@ sub put
# GUID for this protocol as per RFC6455 Section 1.3 # GUID for this protocol as per RFC6455 Section 1.3
my $websocketKeyResponseRaw = $websocketKey."258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; my $websocketKeyResponseRaw = $websocketKey."258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
my $websocketKeyResponse = sha1_base64($websocketKeyResponseRaw); my $websocketKeyResponse = sha1_base64($websocketKeyResponseRaw);
# Pad up to base64 length 4[N/3] # Pad up to base64 length 4[N/3]
$websocketKeyResponse .= "=" x ((length($websocketKeyResponse) * 3) % 4); $websocketKeyResponse .= "=" x ((length($websocketKeyResponse) * 3) % 4);
$response->push_header('Sec-WebSocket-Accept',$websocketKeyResponse); $response->push_header('Sec-WebSocket-Accept',$websocketKeyResponse);
} }
...@@ -141,18 +154,22 @@ sub put ...@@ -141,18 +154,22 @@ sub put
push(@results,$self->_build_raw_response($response)); push(@results,$self->_build_raw_response($response));
} }
# Handle WebSocket data # Handle WebSocket data
} elsif ($self->{'state'} == ST_WEBSOCKET_STREAM) { } elsif ($self->{'state'} == ST_WEBSOCKET_STREAM) {
# Compile our list of results # Compile our list of results
foreach my $response (@{$responses}) { foreach my $response (@{$responses}) {
# If we don't have a websocket write state, create one # If we don't have a websocket state, create one
if (!$self->{'state_websocket_write'}) { if (!$self->{'websocket_state'}) {
$self->{'state_websocket_write'} = new POE::Filter::HybridHTTP::WebSocketFrame(); $self->{'websocket_state'} = new opentrafficshaper::POE::Filter::HybridHTTP::WebSocketFrame();
} }
$self->{'state_websocket_write'}->append($response); # Don't mask replies from server to client RFC6455 secion 5.1.
$self->{'websocket_state'}->masked(0);
push(@results,$self->{'state_websocket_write'}->to_bytes()); # Consume the response with websockets...
$self->{'websocket_state'}->append($response);
# Spit out bytes...
my $payload = $self->{'websocket_state'}->to_bytes();
push(@results,$payload);
} }
} }
...@@ -174,8 +191,7 @@ sub _reset ...@@ -174,8 +191,7 @@ sub _reset
# Reset our filter state # Reset our filter state
$self->{'buffer'} = ''; $self->{'buffer'} = '';
$self->{'state'} = ST_HTTP_HEADERS; $self->{'state'} = ST_HTTP_HEADERS;
$self->{'state_websocket_read'} = undef; $self->{'websocket_state'} = undef;
$self->{'state_websocket_write'} = undef;
$self->{'last_request'} = $self->{'request'}; $self->{'last_request'} = $self->{'request'};
$self->{'request'} = undef; # We want the last request always $self->{'request'} = undef; # We want the last request always
$self->{'content_len'} = 0; $self->{'content_len'} = 0;
...@@ -185,7 +201,7 @@ sub _reset ...@@ -185,7 +201,7 @@ sub _reset
# Internal function to parse an HTTP status line and return the HTTP # Internal function to parse an HTTP status line and return the HTTP
# protocol version. # protocol version.
sub _http_version sub _http_version
{ {
my $version = shift; my $version = shift;
...@@ -214,7 +230,7 @@ sub _get_one_http_headers ...@@ -214,7 +230,7 @@ sub _get_one_http_headers
if ($self->{'buffer'} !~ s/^(\S.*?(?:\r?\n){2})//s) { if ($self->{'buffer'} !~ s/^(\S.*?(?:\r?\n){2})//s) {
return [ ]; return [ ];
} }
# Pull the headers as a string off the buffer # Pull the headers as a string off the buffer
my $header_str = $1; my $header_str = $1;
# Parse the request line. # Parse the request line.
...@@ -258,7 +274,7 @@ sub _get_one_http_headers ...@@ -258,7 +274,7 @@ sub _get_one_http_headers
# We no longer matching, so this is the last header? # We no longer matching, so this is the last header?
} else { } else {
last HEADER; last HEADER;
} }
} }
# Push on the last header if we had one... # Push on the last header if we had one...
$request->push_header($key, $val) if $key; $request->push_header($key, $val) if $key;
...@@ -272,7 +288,7 @@ sub _get_one_http_headers ...@@ -272,7 +288,7 @@ sub _get_one_http_headers
$content_length = int($content_length); $content_length = int($content_length);
} }
my $content_encoding = $request->content_encoding(); my $content_encoding = $request->content_encoding();
# The presence of a message-body in a request is signaled by the # The presence of a message-body in a request is signaled by the
# inclusion of a Content-Length or Transfer-Encoding header field in # inclusion of a Content-Length or Transfer-Encoding header field in
# the request's message-headers. A message-body MUST NOT be included in # the request's message-headers. A message-body MUST NOT be included in
...@@ -306,17 +322,17 @@ sub _get_one_http_headers ...@@ -306,17 +322,17 @@ sub _get_one_http_headers
# the server SHOULD respond with 400 (bad request) if it cannot # the server SHOULD respond with 400 (bad request) if it cannot
# determine the length of the message, or with 411 (length required) if # determine the length of the message, or with 411 (length required) if
# it wishes to insist on receiving a valid Content-Length. # it wishes to insist on receiving a valid Content-Length.
# - RFC2616 # - RFC2616
# PG- This seems to imply that we can either detect the length (but how # PG- This seems to imply that we can either detect the length (but how
# would one do that?) or require a Content-Length header. We do the # would one do that?) or require a Content-Length header. We do the
# latter. # latter.
# #
# PG- Dispite all the above, I'm not fully sure this implements RFC2616 # PG- Dispite all the above, I'm not fully sure this implements RFC2616
# properly. There's something about transfer-coding that I don't fully # properly. There's something about transfer-coding that I don't fully
# understand. # understand.
if (!$content_length) { if (!$content_length) {
# assume a Content-Length of 0 is valid pre 1.1 # assume a Content-Length of 0 is valid pre 1.1
if ($proto >= $HTTP_1_1 && !defined($content_length)) { if ($proto >= $HTTP_1_1 && !defined($content_length)) {
# We have Content-Encoding, but not Content-Length. # We have Content-Encoding, but not Content-Length.
...@@ -328,7 +344,7 @@ sub _get_one_http_headers ...@@ -328,7 +344,7 @@ sub _get_one_http_headers
$self->{'content_length'} = $content_length; $self->{'content_length'} = $content_length;
$self->{'state'} = ST_HTTP_CONTENT; $self->{'state'} = ST_HTTP_CONTENT;
$self->{'request'} = $request; $self->{'request'} = $request;
$self->_get_one_http_content(); $self->_get_one_http_content();
} }
...@@ -387,16 +403,16 @@ sub _get_one_websocket_record ...@@ -387,16 +403,16 @@ sub _get_one_websocket_record
# If we don't have a websocket state, create one # If we don't have a websocket state, create one
if (!$self->{'state_websocket_read'}) { if (!$self->{'websocket_state'}) {
$self->{'state_websocket_read'} = new POE::Filter::HybridHTTP::WebSocketFrame(); $self->{'websocket_state'} = new opentrafficshaper::POE::Filter::HybridHTTP::WebSocketFrame();
} }
$self->{'state_websocket_read'}->append($self->{'buffer'}); $self->{'websocket_state'}->append($self->{'buffer'});
# Blank our buffer # Blank our buffer
$self->{'buffer'} = ''; $self->{'buffer'} = '';
# Loop with records and push onto result set # Loop with records and push onto result set
my @results; my @results;
while (my $item = $self->{'state_websocket_read'}->next()) { while (my $item = $self->{'websocket_state'}->next()) {
push(@results,$item); push(@results,$item);
} }
...@@ -464,7 +480,7 @@ sub _build_raw_response ...@@ -464,7 +480,7 @@ sub _build_raw_response
# Set our content length # Set our content length
# - This is required even if the content length is 0, for instance when we doing a REDIRECT with no content some browsers hang if # - This is required even if the content length is 0, for instance when we doing a REDIRECT with no content some browsers hang if
# - there is no content length set. # - there is no content length set.
$response->push_header("Content-Length",length($response->content)); $response->header("Content-Length" => length($response->content));
# Setup our output # Setup our output
my $output = sprintf("%s %s",$self->{'protocol'},$response->status_line); my $output = sprintf("%s %s",$self->{'protocol'},$response->status_line);
...@@ -478,3 +494,4 @@ sub _build_raw_response ...@@ -478,3 +494,4 @@ sub _build_raw_response
1; 1;
# vim: ts=4
# POE::Filter::HybridHTTP::WebSocketFrame - Copyright 2013, AllworldIT # POE::Filter::HybridHTTP::WebSocketFrame - Copyright 2007-2023, AllworldIT
# Hybrid HTTP filter support for WebSocketFrames # Hybrid HTTP filter support for WebSocketFrames
#
# 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/>.
## ##
# Code originally based on Protocol::WebSocket::Frame # Code originally based on Protocol::WebSocket::Frame
...@@ -20,7 +33,9 @@ ...@@ -20,7 +33,9 @@
# the same terms as Perl 5.10. # the same terms as Perl 5.10.
## ##
package POE::Filter::HybridHTTP::WebSocketFrame; package opentrafficshaper::POE::Filter::HybridHTTP::WebSocketFrame;
use bytes;
use strict; use strict;
use warnings; use warnings;
...@@ -95,7 +110,7 @@ sub next { ...@@ -95,7 +110,7 @@ sub next {
return Encode::decode('UTF-8', $bytes); return Encode::decode('UTF-8', $bytes);
} }
return; return;
} }
sub fin { @_ > 1 ? $_[0]->{fin} = $_[1] : $_[0]->{fin} } sub fin { @_ > 1 ? $_[0]->{fin} = $_[1] : $_[0]->{fin} }
...@@ -133,7 +148,7 @@ sub next_bytes { ...@@ -133,7 +148,7 @@ sub next_bytes {
$offset += 1; # FIN,RSV[1-3],OPCODE $offset += 1; # FIN,RSV[1-3],OPCODE
# Grab payload length # Grab payload length
my $payload_len = unpack 'C', substr($self->{'buffer'}, $offset, 1); my $payload_len = unpack('C',substr($self->{'buffer'}, $offset, 1));
# Check if the payload is masked, if it is flag it internally # Check if the payload is masked, if it is flag it internally
my $masked = ($payload_len & 0b10000000) >> 7; my $masked = ($payload_len & 0b10000000) >> 7;
...@@ -147,7 +162,7 @@ sub next_bytes { ...@@ -147,7 +162,7 @@ sub next_bytes {
return; return;
} }
# Unpack the payload_len into its actual value & bump the offset # Unpack the payload_len into its actual value & bump the offset
$payload_len = unpack 'n', substr($self->{'buffer'}, $offset, 2); $payload_len = unpack('n',substr($self->{'buffer'},$offset,2));
$offset += 2; $offset += 2;
} elsif ($payload_len > 126) { } elsif ($payload_len > 126) {
...@@ -174,7 +189,7 @@ sub next_bytes { ...@@ -174,7 +189,7 @@ sub next_bytes {
$offset += 8; $offset += 8;
} }
# XXX - not sure how to return this sanely # XXX - not sure how to return this sanely
if ($payload_len > $self->{'max_payload_size'}) { if ($payload_len > $self->{'max_payload_size'}) {
$self->{'buffer'} = ''; $self->{'buffer'} = '';
return; return;
} }
...@@ -245,6 +260,7 @@ sub next_bytes { ...@@ -245,6 +260,7 @@ sub next_bytes {
return; return;
} }
sub to_bytes { sub to_bytes {
my $self = shift; my $self = shift;
...@@ -259,38 +275,40 @@ sub to_bytes { ...@@ -259,38 +275,40 @@ sub to_bytes {
$opcode = $self->opcode || 1; $opcode = $self->opcode || 1;
} }
$string .= pack 'C', ($opcode + 128); # Set FIN + black RSV + set OPCODE in the first 8 bites
$string .= pack('C',($opcode | 0b10000000) & 0b10001111);
my $payload_len = length($self->{'buffer'}); my $payload_len = length($self->{'buffer'});
if ($payload_len <= 125) { if ($payload_len <= 125) {
# Flip masked bit if we're masked
$payload_len |= 0b10000000 if $self->masked; $payload_len |= 0b10000000 if $self->masked;
$string .= pack 'C', $payload_len; # Encode the payload length and add to string
$string .= pack('C',$payload_len);
} }
elsif ($payload_len <= 0xffff) { elsif ($payload_len <= 0xffff) {
$string .= pack 'C', 126 + ($self->masked ? 128 : 0); my $bits = 0b01111110;
$string .= pack 'n', $payload_len; $bits |= 0b10000000 if $self->masked;
$string .= pack('C',$bits);
$string .= pack('n',$payload_len);
} }
else { else {
$string .= pack 'C', 127 + ($self->masked ? 128 : 0); my $bits = 0b01111111;
$bits |= 0b10000000 if $self->masked;
$string .= pack('C',$bits);
# Shifting by an amount >= to the system wordsize is undefined # Shifting by an amount >= to the system wordsize is undefined
$string .= pack 'N', $Config{'ivsize'} <= 4 ? 0 : $payload_len >> 32; $string .= pack('N',$Config{'ivsize'} <= 4 ? 0 : $payload_len >> 32);
$string .= pack 'N', ($payload_len & 0xffffffff); $string .= pack('N',($payload_len & 0xffffffff));
} }
if ($self->masked) { if ($self->masked) {
my $mask = $self->{mask} || ( MATH_RANDOM_SECURE ? Math::Random::Secure::irand(MAX_RAND_INT) : int(rand(MAX_RAND_INT)) );
my $mask = $self->{mask} $mask = pack('N',$mask);
|| (
MATH_RANDOM_SECURE
? Math::Random::Secure::irand(MAX_RAND_INT)
: int(rand(MAX_RAND_INT))
);
$mask = pack 'N', $mask;
$string .= $mask; $string .= $mask;
$string .= $self->_mask($self->{'buffer'}, $mask); $string .= $self->_mask($self->{'buffer'},$mask);
} }
else { else {
$string .= $self->{'buffer'}; $string .= $self->{'buffer'};
...@@ -302,6 +320,7 @@ sub to_bytes { ...@@ -302,6 +320,7 @@ sub to_bytes {
return $string; return $string;
} }
sub _mask { sub _mask {
my $self = shift; my $self = shift;
my ($payload, $mask) = @_; my ($payload, $mask) = @_;
...@@ -314,3 +333,4 @@ sub _mask { ...@@ -314,3 +333,4 @@ sub _mask {
} }
1; 1;
# vim: ts=4
# OpenTrafficShaper POE::Filter::TCStatistics TC stats filter # OpenTrafficShaper POE::Filter::TCStatistics TC stats filter
# OpenTrafficShaper webserver module: limits page # OpenTrafficShaper webserver module: limits page
# Copyright (C) 2007-2013, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -30,28 +30,27 @@ ...@@ -30,28 +30,27 @@
# and from HTTPD filters, they should submit their request as a patch. # and from HTTPD filters, they should submit their request as a patch.
## ##
package POE::Filter::TCStatistics; package opentrafficshaper::POE::Filter::TCStatistics;
use warnings; use warnings;
use strict; use strict;
use bytes; use JSON qw(decode_json);
use POE::Filter; use POE::Filter;
use vars qw($VERSION @ISA); use vars qw($VERSION @ISA);
# NOTE - Should be #.### (three decimal places) # NOTE - Should be #.### (three decimal places)
$VERSION = '1.000'; $VERSION = '3.000';
@ISA = qw(POE::Filter); @ISA = qw(POE::Filter);
# Class instantiation # Class instantiation
sub new sub new
{ {
my $class = shift; my $class = shift;
# These are our internal properties # These are our internal properties
my $self = { }; my $self = { };
# Build our class # Build our class
bless($self, $class); bless($self, $class);
...@@ -63,10 +62,11 @@ sub new ...@@ -63,10 +62,11 @@ sub new
} }
# From the docs: # From the docs:
# get_one_start() accepts an array reference containing unprocessed stream chunks. The chunks are added to the filter's Internal # get_one_start() accepts an array reference containing unprocessed stream chunks. The chunks are added to the filter's Internal
# buffer for parsing by get_one(). # buffer for parsing by get_one().
sub get_one_start sub get_one_start
{ {
my ($self, $stream) = @_; my ($self, $stream) = @_;
...@@ -78,71 +78,75 @@ sub get_one_start ...@@ -78,71 +78,75 @@ sub get_one_start
} }
# This is called to see if we can grab records/items # This is called to see if we can grab records/items
sub get_one sub get_one
{ {
my $self = shift; my $self = shift;
my @results = (); my @results = ();
# Pull of blocks of class info's # If we have a empty buffer we can just return
while ($self->{'buffer'} =~ s/^(class.+)\n\s+(.+\n\s+.+)\n.+\n.+\n\n//m) { return [] if ($self->{'buffer'} eq '');
my $curstat; chomp($self->{'buffer'});
# NK: It seems we may not get the terminating ] at the end if the output is still busy?
return [] if (substr($self->{'buffer'},-1) ne "]");
my ($classStr,$statsStr) = ($1,$2);
# Strip off the line into an array # Try decode JSON payload
my @classArray = split(/\s+/,$classStr); my $items;
my @statsArray = split(/[\s,\(\)]+/,$statsStr); eval {
# Pull in all the items $items = decode_json($self->{'buffer'});
# class htb 1:1 root rate 100000Kbit ceil 100000Kbit burst 51800b cburst 51800b $self->{'buffer'} = '';
if (@classArray == 12) { 1;
$curstat->{'cir'} = _getKNumber($classArray[5]); } or do {
$curstat->{'limit'} = _getKNumber($classArray[7]); print(STDERR "FAILED TO DECODE JSON: >" . $self->{'buffer'} . "<\n");
# class htb 1:d parent 1:1 rate 10000Kbit ceil 100000Kbit burst 6620b cburst 51800b return [];
} elsif (@classArray == 13) { };
$curstat->{'cir'} = _getKNumber($classArray[6]);
$curstat->{'limit'} = _getKNumber($classArray[8]); # Loop with each item and generate a current stat
# class htb 1:3 parent 1:1 prio 7 rate 10000Kbit ceil 100000Kbit burst 6620b cburst 51800b for my $item (@{$items}) {
} elsif (@classArray == 15) { my $curstat = {};
$curstat->{'priority'} = int($classArray[6]);
$curstat->{'cir'} = _getKNumber($classArray[8]); # Skip everything except HTB
$curstat->{'limit'} = _getKNumber($classArray[10]); if ($item->{'class'} ne "htb") {
# class htb 1:3 parent 1:1 leaf 3: prio 7 rate 10000Kbit ceil 100000Kbit burst 6620b cburst 51800b continue;
} elsif (@classArray == 17) {
$curstat->{'priority'} = int($classArray[8]);
$curstat->{'cir'} = _getKNumber($classArray[10]);
$curstat->{'limit'} = _getKNumber($classArray[12]);
} else {
next;
}
($curstat->{'_class_parent'},$curstat->{'_class_child'}) = split(/:/,$classArray[2]);
# Sent 0 bytes 0 pkt (dropped 0, overlimits 0 requeues 0)
# rate 0bit 0pps backlog 0b 0p requeues 0
if (@statsArray == 19) {
$curstat->{'total_bytes'} = int($statsArray[1]);
$curstat->{'total_packets'} = int($statsArray[3]);
$curstat->{'total_dropped'} = int($statsArray[6]);
$curstat->{'total_overlimits'} = int($statsArray[8]);
$curstat->{'rate'} = _getKNumber($statsArray[12]);
$curstat->{'pps'} = int(substr($statsArray[13],0,-3));
$curstat->{'queue_size'} = int(substr($statsArray[15],0,-1));
$curstat->{'queue_len'} = int(substr($statsArray[16],0,-1));
} else {
next;
} }
push(@results,$curstat); # Split off the handle into the parent and child
( $curstat->{'TCClassParent'}, $curstat->{'TCClassChild'} ) = split( /:/, $item->{'handle'} );
# The rate and ceil is represented in bytes/s, so it needs to be multiplied by 8 and divided by 1000 to get Kbit/s
$curstat->{'CIR'} = (int($item->{'rate'}) * 8) / 1000;
$curstat->{'Limit'} = (int($item->{'ceil'}) * 8) / 1000;
# If we have a prio, we need to add this too
if (defined($item->{'prio'})) {
$curstat->{'Priority'} = int($item->{'prio'});
}
# We should probably always have these
$curstat->{'TotalBytes'} = int($item->{'stats'}->{'bytes'}) // 0;
$curstat->{'TotalPackets'} = int($item->{'stats'}->{'packets'}) // 0;
$curstat->{'TotalDropped'} = int($item->{'stats'}->{'drops'}) // 0;
$curstat->{'TotalOverLimits'} = int($item->{'stats'}->{'overlimits'}) // 0;
$curstat->{'QueueSize'} = int( $item->{'stats'}->{'backlog'} ) // 0;
$curstat->{'QueueLen'} = int( $item->{'stats'}->{'qlen'} ) // 0;
# These are HTB specific if stats are enabled
$curstat->{'Rate'} = 0;
if (defined($item->{'stats'}->{'rate'})) {
$curstat->{'Rate'} = (int($item->{'stats'}->{'rate'}) * 8) / 1000;
}
$curstat->{'PPS'} = 0;
if (defined($item->{'stats'}->{'pps'})) {
$curstat->{'PPS'} = int($item->{'stats'}->{'pps'});
}
push(@results,$curstat);
} }
return [ @results ]; return [ @results ];
} }
# Function to push data to the socket # Function to push data to the socket
sub put sub put
{ {
my ($self, $data) = @_; my ($self, $data) = @_;
...@@ -168,6 +172,7 @@ sub _reset ...@@ -168,6 +172,7 @@ sub _reset
} }
# Get rate... # Get rate...
sub _getKNumber sub _getKNumber
{ {
...@@ -181,9 +186,9 @@ sub _getKNumber ...@@ -181,9 +186,9 @@ sub _getKNumber
} elsif ($multiplier eq "K") { } elsif ($multiplier eq "K") {
# noop # noop
} elsif ($multiplier eq "M") { } elsif ($multiplier eq "M") {
$num *= 1000000; $num *= 1000;
} elsif ($multiplier eq "G") { } elsif ($multiplier eq "G") {
$num *= 1000000000; $num *= 1000000;
} }
return int($num); return int($num);
......
awitpt @ 932340ef
Subproject commit 932340ef7336dc40bdd3d35d2023bde1310e9066
# OpenTrafficShaper constants package # OpenTrafficShaper constants package
# Copyright (C) 2013-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
......
# Logging functionality # Logging functionality
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -49,6 +49,7 @@ use IO::Handle; ...@@ -49,6 +49,7 @@ use IO::Handle;
use POSIX qw( strftime ); use POSIX qw( strftime );
# Instantiate # Instantiate
sub new { sub new {
my ($class) = @_; my ($class) = @_;
...@@ -60,6 +61,8 @@ sub new { ...@@ -60,6 +61,8 @@ sub new {
return $self; return $self;
} }
# Logging function # Logging function
sub log sub log
{ {
...@@ -97,6 +100,8 @@ sub log ...@@ -97,6 +100,8 @@ sub log
} }
} }
# Set log file & open it # Set log file & open it
sub open sub open
{ {
...@@ -113,6 +118,8 @@ sub open ...@@ -113,6 +118,8 @@ sub open
$self->{'handle'} = $fh; $self->{'handle'} = $fh;
} }
# Set log level # Set log level
sub setLevel sub setLevel
{ {
...@@ -123,5 +130,7 @@ sub setLevel ...@@ -123,5 +130,7 @@ sub setLevel
$self->{'level'} = $level; $self->{'level'} = $level;
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper Plugin Handler # OpenTrafficShaper Plugin Handler
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -38,6 +38,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -38,6 +38,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
my $globals; my $globals;
# Check if a plugin is loaded # Check if a plugin is loaded
sub isPluginLoaded sub isPluginLoaded
{ {
...@@ -48,6 +49,7 @@ sub isPluginLoaded ...@@ -48,6 +49,7 @@ sub isPluginLoaded
} }
# Function to register a plugin # Function to register a plugin
sub plugin_register sub plugin_register
{ {
...@@ -80,8 +82,10 @@ sub plugin_register ...@@ -80,8 +82,10 @@ sub plugin_register
# Check if the error is critical or not # Check if the error is critical or not
if ($systemPlugin) { if ($systemPlugin) {
$logger->log(LOG_ERR,"[PLUGINS] Error loading plugin '%s', things WILL BREAK! (%s)",$pluginName,$@); $logger->log(LOG_ERR,"[PLUGINS] Error loading plugin '%s', things WILL BREAK! (%s)",$pluginName,$@);
exit;
} else { } else {
$logger->log(LOG_WARN,"[PLUGINS] Error loading plugin '%s' (%s)",$pluginName,$@); $logger->log(LOG_WARN,"[PLUGINS] Error loading plugin '%s' (%s)",$pluginName,$@);
exit;
} }
} else { } else {
$logger->log(LOG_DEBUG,"[PLUGINS] Plugin '%s' loaded.",$pluginName); $logger->log(LOG_DEBUG,"[PLUGINS] Plugin '%s' loaded.",$pluginName);
...@@ -90,6 +94,7 @@ sub plugin_register ...@@ -90,6 +94,7 @@ sub plugin_register
} }
# Setup our main config ref # Setup our main config ref
sub init sub init
{ {
...@@ -151,5 +156,6 @@ sub _plugin_register { ...@@ -151,5 +156,6 @@ sub _plugin_register {
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper configuration manager # OpenTrafficShaper configuration manager
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -21,16 +21,29 @@ package opentrafficshaper::plugins::configmanager; ...@@ -21,16 +21,29 @@ package opentrafficshaper::plugins::configmanager;
use strict; use strict;
use warnings; use warnings;
use POE; use POE;
use Storable qw( dclone ); use Storable qw(
dclone
);
use Time::HiRes qw(
gettimeofday
tv_interval
);
use awitpt::util qw(
isNumber ISNUMBER_ALLOW_ZERO
isUsername ISUSERNAME_ALLOW_ATSIGN
prettyUndef
getHashChanges
);
use opentrafficshaper::constants; use opentrafficshaper::constants;
use opentrafficshaper::logger; use opentrafficshaper::logger;
use opentrafficshaper::utils; use opentrafficshaper::util qw(
isIPv46 isIPv46CIDR
);
# NK: TODO: Maybe we want to remove timing at some stage? maybe not?
use Time::HiRes qw( gettimeofday tv_interval );
# Exporter stuff # Exporter stuff
...@@ -40,13 +53,31 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -40,13 +53,31 @@ our (@ISA,@EXPORT,@EXPORT_OK);
@EXPORT = qw( @EXPORT = qw(
); );
@EXPORT_OK = qw( @EXPORT_OK = qw(
createGroup
isGroupIDValid
createTrafficClass
getTrafficClass
getTrafficClasses
getInterfaceTrafficClass
getAllTrafficClasses
isTrafficClassIDValid
isInterfaceIDValid
createInterface
createInterfaceClass
createInterfaceGroup
changeInterfaceTrafficClass
getEffectiveInterfaceTrafficClass2
isInterfaceTrafficClassValid
setInterfaceTrafficClassShaperState
unsetInterfaceTrafficClassShaperState
createLimit createLimit
getLimit
getLimits
getLimitUsername
getOverride getPoolOverride
getOverrides getPoolOverrides
createPool createPool
removePool removePool
...@@ -64,6 +95,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -64,6 +95,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
setPoolShaperState setPoolShaperState
unsetPoolShaperState unsetPoolShaperState
isPoolIDValid isPoolIDValid
isPoolOverridden
isPoolReady isPoolReady
getEffectivePool getEffectivePool
...@@ -85,23 +117,14 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -85,23 +117,14 @@ our (@ISA,@EXPORT,@EXPORT_OK);
removePoolMemberAttribute removePoolMemberAttribute
isPoolMemberReady isPoolMemberReady
getTrafficClasses
getAllTrafficClasses
getTrafficClassName
isTrafficClassIDValid
getTrafficClassPriority getTrafficClassPriority
getTrafficDirection getTrafficDirection
isInterfaceIDValid
isGroupIDValid
getInterface getInterface
getInterfaces getInterfaces
getInterfaceTrafficClasses
getInterfaceDefaultPool getInterfaceDefaultPool
getInterfaceRate getInterfaceLimit
getInterfaceGroup getInterfaceGroup
getInterfaceGroups getInterfaceGroups
isInterfaceGroupIDValid isInterfaceGroupIDValid
...@@ -111,10 +134,10 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -111,10 +134,10 @@ our (@ISA,@EXPORT,@EXPORT_OK);
); );
use constant { use constant {
VERSION => '0.0.1', VERSION => '1.0.0',
# After how long does a limit get removed if its's deemed offline # After how long does a limit get removed if its's deemed offline
TIMEOUT_EXPIRE_OFFLINE => 300, TIMEOUT_EXPIRE_OFFLINE => 10,
# How often our config check ticks # How often our config check ticks
TICK_PERIOD => 5, TICK_PERIOD => 5,
...@@ -130,7 +153,7 @@ sub POOL_REQUIRED_ATTRIBUTES { ...@@ -130,7 +153,7 @@ sub POOL_REQUIRED_ATTRIBUTES {
qw( qw(
Name Name
InterfaceGroupID InterfaceGroupID
ClassID TrafficLimitTx TrafficLimitRx TrafficClassID TxCIR RxCIR
Source Source
) )
} }
...@@ -139,7 +162,7 @@ sub POOL_REQUIRED_ATTRIBUTES { ...@@ -139,7 +162,7 @@ sub POOL_REQUIRED_ATTRIBUTES {
sub POOL_CHANGE_ATTRIBUTES { sub POOL_CHANGE_ATTRIBUTES {
qw( qw(
FriendlyName FriendlyName
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
Expires Expires
Notes Notes
) )
...@@ -152,13 +175,35 @@ sub POOL_PERSISTENT_ATTRIBUTES { ...@@ -152,13 +175,35 @@ sub POOL_PERSISTENT_ATTRIBUTES {
Name Name
FriendlyName FriendlyName
InterfaceGroupID InterfaceGroupID
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
Expires Expires
Source Source
Notes Notes
) )
} }
# Class attributes that can be changed (overridden)
sub CLASS_CHANGE_ATTRIBUTES {
qw(
CIR Limit
)
}
# Class attributes that can be overidden
sub CLASS_OVERRIDE_CHANGESET_ATTRIBUTES {
qw(
CIR Limit
)
}
# Class attributes that can be overidden
sub CLASS_OVERRIDE_PERSISTENT_ATTRIBUTES {
qw(
InterfaceID
TrafficClassID
CIR Limit
)
}
# Mandatory pool member attributes # Mandatory pool member attributes
sub POOLMEMBER_REQUIRED_ATTRIBUTES { sub POOLMEMBER_REQUIRED_ATTRIBUTES {
...@@ -185,6 +230,8 @@ sub POOLMEMBER_PERSISTENT_ATTRIBUTES { ...@@ -185,6 +230,8 @@ sub POOLMEMBER_PERSISTENT_ATTRIBUTES {
qw( qw(
FriendlyName FriendlyName
Username IPAddress Username IPAddress
IPNATAddress
IPNATInbound
MatchPriorityID MatchPriorityID
PoolID PoolID
GroupID GroupID
...@@ -201,54 +248,54 @@ sub LIMIT_REQUIRED_ATTRIBUTES { ...@@ -201,54 +248,54 @@ sub LIMIT_REQUIRED_ATTRIBUTES {
Username IPAddress Username IPAddress
InterfaceGroupID MatchPriorityID InterfaceGroupID MatchPriorityID
GroupID GroupID
ClassID TrafficLimitTx TrafficLimitRx TrafficClassID TxCIR RxCIR
Source Source
) )
} }
# Override match attributes, one is required # Pool override match attributes, one is required
sub OVERRIDE_MATCH_ATTRIBUTES { sub POOL_OVERRIDE_MATCH_ATTRIBUTES {
qw( qw(
PoolName Username IPAddress PoolName Username IPAddress
GroupID GroupID
) )
} }
# Override attributes # Pool override attributes
sub OVERRIDE_ATTRIBUTES { sub POOL_OVERRIDE_ATTRIBUTES {
qw( qw(
FriendlyName FriendlyName
PoolName Username IPAddress GroupID PoolName Username IPAddress GroupID
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
Expires Expires
Notes Notes
) )
} }
# Override attributes that can be changed # Pool override attributes that can be changed
sub OVERRIDE_CHANGE_ATTRIBUTES { sub POOL_OVERRIDE_CHANGE_ATTRIBUTES {
qw( qw(
FriendlyName FriendlyName
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
Expires Expires
Notes Notes
) )
} }
# Override changeset attributes # Pool override changeset attributes
sub OVERRIDE_CHANGESET_ATTRIBUTES { sub POOL_OVERRIDE_CHANGESET_ATTRIBUTES {
qw( qw(
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
) )
} }
# Override attributes supported for persistent storage # Pool override attributes supported for persistent storage
sub OVERRIDE_PERSISTENT_ATTRIBUTES { sub POOL_OVERRIDE_PERSISTENT_ATTRIBUTES {
qw( qw(
FriendlyName FriendlyName
PoolName Username IPAddress GroupID PoolName Username IPAddress GroupID
ClassID TrafficLimitTx TrafficLimitRx TrafficLimitTxBurst TrafficLimitRxBurst TrafficClassID TxCIR RxCIR TxLimit RxLimit
Notes Notes
Expires Created Expires Created
Source Source
...@@ -269,29 +316,13 @@ our $pluginInfo = { ...@@ -269,29 +316,13 @@ our $pluginInfo = {
}; };
# Copy of system globals # This modules globals
my $globals; my $globals;
# System logger
my $logger; my $logger;
# Configuration for this plugin # Configuration for this plugin
our $config = { our $config = {
# Class to use for unclassified traffic
'default_pool' => undef,
# Traffic groups
'groups' => {
1 => 'Default'
},
# Traffic classes
'classes' => {
1 => 'Default'
},
# Interfaces
'interfaces' => {
},
# Interface groups
'interface_groups' => {
},
# Match priorities # Match priorities
'match_priorities' => { 'match_priorities' => {
1 => 'First', 1 => 'First',
...@@ -302,143 +333,202 @@ our $config = { ...@@ -302,143 +333,202 @@ our $config = {
'statefile' => '/var/lib/opentrafficshaper/configmanager.state', 'statefile' => '/var/lib/opentrafficshaper/configmanager.state',
}; };
# Last time the cleanup ran
my $lastCleanup = time(); #
# If our state has changed and when last we sync'd to disk # GROUPS - pool members are linked to groups
my $stateChanged = 0; #
my $lastStateSync = time(); # Attributes:
# * ID
# * Name
#
# $globals->{'Groups'}
#
# CLASSES
#
# Attributes:
# * ID
# * Name
#
# $globals->{'TrafficClasses'}
# #
# INTERFACES # INTERFACES
# #
my $interfaceIPMap = {}; # Attributes:
# * ID
# * Name
# * Interface
# * Limit
#
# $globals->{'Interfaces'}
# #
# POOLS # POOLS
# #
# Parameters: # Parameters:
# * ID
# * FriendlyName # * FriendlyName
# - Used for display purposes # - Used for display purposes
# * Name # * Name
# - Unix timestamp when this entry expires, 0 if never # - Unix timestamp when this entry expires, 0 if never
# * ClassID # * TrafficClassID
# - Class ID # - Traffic class ID
# * InterfaceGroupID # * InterfaceGroupID
# - Interface group this pool is attached to # - Interface group this pool is attached to
# * TrafficLimitTx # * TxCIR
# - Traffic limit in kbps # - Traffic limit in kbps
# * TrafficLimitRx # * RxCIR
# - Traffic limit in kbps # - Traffic limit in kbps
# * TrafficLimitTxBurst # * TxLimit
# - Traffic bursting limit in kbps # - Traffic bursting limit in kbps
# * TrafficLimitRxBurst # * RxLimit
# - Traffic bursting limit in kbps # - Traffic bursting limit in kbps
# * Notes # * Notes
# - Notes on this limit # - Notes on this limit
# * Source # * Source
# - This is the source of the limit, typically plugin.ModuleName # - This is the source of the limit, typically plugin.ModuleName
my $pools = { }; #
my $poolNameMap = { }; # $globals->{'Pools'}
my $poolIDCounter = 1; # $globals->{'PoolNameMap'}
# $globals->{'PoolIDCounter'}
# #
# POOL MEMBERS # POOL MEMBERS
# #
# Supoprted user attributes: # Supoprted user attributes:
# * ID
# * PoolID # * PoolID
# - Pool ID # - Pool ID
# * Username # * Username
# - Users username # - Users username
# * IPAddress # * IPAddress
# - Users IP address # - Users IP address
# * GroupID # * GroupID
# - Group ID # - Group ID
# * MatchPriorityID # * MatchPriorityID
# - Match priority on the backend of this limit # - Match priority on the backend of this limit
# * ClassID # * TrafficClassID
# - Class ID # - Class ID
# * Expires # * Expires
# - Unix timestamp when this entry expires, 0 if never # - Unix timestamp when this entry expires, 0 if never
# * FriendlyName # * FriendlyName
# - Used for display purposes instead of username if specified # - Used for display purposes instead of username if specified
# * Notes # * Notes
# - Notes on this limit # - Notes on this limit
# * Status # * Status
# - new # - new
# - offline # - offline
# - online # - online
# - unknown # - unknown
# * Source # * Source
# - This is the source of the limit, typically plugin.ModuleName # - This is the source of the limit, typically plugin.ModuleName
my $poolMembers = { }; #
my $poolMemberIDCounter = 1; # $globals->{'PoolMembers'}
my $poolMemberMap = { }; # $globals->{'PoolMemberIDCounter'}
# $globals->{'PoolMemberMap'}
# #
# OVERRIDES # POOL OVERRIDES
# #
# Selection criteria: # Selection criteria:
# * PoolName # * PoolName
# - Pool name # - Pool name
# * Username # * Username
# - Users username # - Users username
# * IPAddress # * IPAddress
# - Users IP address # - Users IP address
# * GroupID # * GroupID
# - Group ID # - Group ID
# #
# Overrides: # Pool Overrides:
# * ClassID # * TrafficClassID
# - Class ID # - Class ID
# * TrafficLimitTx # * TxCIR
# - Traffic limit in kbps # - Traffic limit in kbps
# * TrafficLimitRx # * RxCIR
# - Traffic limit in kbps # - Traffic limit in kbps
# * TrafficLimitTxBurst # * TxLimit
# - Traffic bursting limit in kbps # - Traffic bursting limit in kbps
# * TrafficLimitRxBurst # * RxLimit
# - Traffic bursting limit in kbps # - Traffic bursting limit in kbps
# #
# Parameters: # Parameters:
# * ID
# * FriendlyName # * FriendlyName
# - Used for display purposes # - Used for display purposes
# * Expires # * Expires
# - Unix timestamp when this entry expires, 0 if never # - Unix timestamp when this entry expires, 0 if never
# * Notes # * Notes
# - Notes on this limit # - Notes on this limit
# * Source # * Source
# - This is the source of the limit, typically plugin.ModuleName # - This is the source of the limit, typically plugin.ModuleName
my $overrides = { }; #
my $overrideIDCounter = 1; # $globals->{'PoolOverrides'}
# $globals->{'PoolOverrideIDCounter'}
# Global change queues #
my $poolChangeQueue = { }; # CHANGE QUEUES
my $poolMemberChangeQueue = { }; #
# $globals->{'PoolChangeQueue'}
# $globals->{'PoolMemberChangeQueue'}
# Initialize plugin # Initialize plugin
sub plugin_init sub plugin_init
{ {
$globals = shift; my $system = shift;
my $now = time();
# Setup our environment # Setup our environment
$logger = $globals->{'logger'}; $logger = $system->{'logger'};
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] OpenTrafficShaper Config Manager v%s - Copyright (c) 2007-2014, AllWorldIT", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] OpenTrafficShaper Config Manager v%s - Copyright (c) 2007-2014, AllWorldIT",
VERSION VERSION
); );
# Initialize
$globals->{'LastCleanup'} = $now;
$globals->{'StateChanged'} = 0;
$globals->{'LastStateSync'} = $now;
$globals->{'Groups'} = { };
$globals->{'TrafficClasses'} = { };
$globals->{'Interfaces'} = { };
$globals->{'InterfaceGroups'} = { };
$globals->{'Pools'} = { };
$globals->{'PoolNameMap'} = { };
$globals->{'PoolIDCounter'} = 1;
$globals->{'DefaultPool'} = undef;
$globals->{'PoolMembers'} = { };
$globals->{'PoolMemberIDCounter'} = 1;
$globals->{'PoolMemberMap'} = { };
$globals->{'PoolOverrides'} = { };
$globals->{'PoolOverrideIDCounter'} = 1;
$globals->{'InterfaceTrafficClasses'} = { };
$globals->{'InterfaceTrafficClassCounter'} = 1;
$globals->{'PoolChangeQueue'} = { };
$globals->{'PoolMemberChangeQueue'} = { };
$globals->{'InterfaceTrafficClassChangeQueue'} = { };
# If we have global config, use it # If we have global config, use it
my $gconfig = { }; my $gconfig = { };
if (defined($globals->{'file.config'}->{'shaping'})) { if (defined($system->{'file.config'}->{'shaping'})) {
$gconfig = $globals->{'file.config'}->{'shaping'}; $gconfig = $system->{'file.config'}->{'shaping'};
} }
# Split off groups to load # Split off groups to load
...@@ -469,8 +559,15 @@ sub plugin_init ...@@ -469,8 +559,15 @@ sub plugin_init
$logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic group definition '%s' has invalid name, ignoring",$group); $logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic group definition '%s' has invalid name, ignoring",$group);
next; next;
} }
$config->{'groups'}->{$groupID} = $groupName; # Create group
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded traffic group '%s' with ID %s.",$groupName,$groupID); $groupID = createGroup({
'ID' => $groupID,
'Name' => $groupName
});
if (defined($groupID)) {
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded traffic group '%s' [%s]",$groupName,$groupID);
}
} }
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Traffic groups loaded"); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Traffic groups loaded");
...@@ -494,8 +591,8 @@ sub plugin_init ...@@ -494,8 +591,8 @@ sub plugin_init
# Skip comments # Skip comments
next if ($class =~ /^\s*#/); next if ($class =~ /^\s*#/);
# Split off class ID and class name # Split off class ID and class name
my ($classID,$className) = split(/:/,$class); my ($trafficClassID,$className) = split(/:/,$class);
if (!defined(isNumber($classID))) { if (!defined(isNumber($trafficClassID))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic class definition '%s' has invalid ID, ignoring",$class); $logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic class definition '%s' has invalid ID, ignoring",$class);
next; next;
} }
...@@ -503,8 +600,17 @@ sub plugin_init ...@@ -503,8 +600,17 @@ sub plugin_init
$logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic class definition '%s' has invalid name, ignoring",$class); $logger->log(LOG_WARN,"[CONFIGMANAGER] Traffic class definition '%s' has invalid name, ignoring",$class);
next; next;
} }
$config->{'classes'}->{$classID} = $className; # Create class
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded traffic class '%s' with ID %s",$className,$classID); $trafficClassID = createTrafficClass({
'ID' => $trafficClassID,
'Name' => $className
});
if (!defined($trafficClassID)) {
next;
}
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded traffic class '%s' [%s]",$className,$trafficClassID);
} }
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Traffic classes loaded"); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Traffic classes loaded");
...@@ -512,8 +618,8 @@ sub plugin_init ...@@ -512,8 +618,8 @@ sub plugin_init
# Load interfaces # Load interfaces
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interfaces..."); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interfaces...");
my @interfaces; my @interfaces;
if (defined($globals->{'file.config'}->{'shaping.interface'})) { if (defined($system->{'file.config'}->{'shaping.interface'})) {
@interfaces = keys %{$globals->{'file.config'}->{'shaping.interface'}}; @interfaces = keys %{$system->{'file.config'}->{'shaping.interface'}};
} else { } else {
@interfaces = ( "eth0", "eth1" ); @interfaces = ( "eth0", "eth1" );
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No interfaces defined, using 'eth0' and 'eth1'"); $logger->log(LOG_NOTICE,"[CONFIGMANAGER] No interfaces defined, using 'eth0' and 'eth1'");
...@@ -523,37 +629,45 @@ sub plugin_init ...@@ -523,37 +629,45 @@ sub plugin_init
# This is the interface config to make things easier for us # This is the interface config to make things easier for us
my $iconfig = { }; my $iconfig = { };
# Check if its defined # Check if its defined
if (defined($globals->{'file.config'}->{'shaping.interface'}) && if (defined($system->{'file.config'}->{'shaping.interface'}) &&
defined($globals->{'file.config'}->{'shaping.interface'}->{$interface}) defined($system->{'file.config'}->{'shaping.interface'}->{$interface})
) { ) {
$iconfig = $globals->{'file.config'}->{'shaping.interface'}->{$interface} $iconfig = $system->{'file.config'}->{'shaping.interface'}->{$interface}
} }
# Check our friendly name for this interface # Check our friendly name for this interface
my $interfaceName = "$interface (auto)";
if (defined($iconfig->{'name'}) && $iconfig->{'name'} ne "") { if (defined($iconfig->{'name'}) && $iconfig->{'name'} ne "") {
$config->{'interfaces'}->{$interface}->{'name'} = $iconfig->{'name'}; $interfaceName = $iconfig->{'name'};
} else { } else {
$config->{'interfaces'}->{$interface}->{'name'} = "$interface (auto)";
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' has no 'name' attribute, using '%s (auto)'", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' has no 'name' attribute, using '%s (auto)'",
$interface,$interface $interface,$interface
); );
} }
# Check our interface rate # Check our interface rate
my $interfaceLimit = 100000;
if (defined($iconfig->{'rate'}) && $iconfig->{'rate'} ne "") { if (defined($iconfig->{'rate'}) && $iconfig->{'rate'} ne "") {
# Check rate is valid # Check limit is valid
if (defined(my $rate = isNumber($iconfig->{'rate'}))) { if (defined(my $rate = isNumber($iconfig->{'rate'}))) {
$config->{'interfaces'}->{$interface}->{'rate'} = $rate; $interfaceLimit = $rate;
} else { } else {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' has invalid 'rate' attribute, using 100000 instead", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' has invalid 'rate' attribute, using 100000 instead",
$interface $interface
); );
} }
} else { } else {
$config->{'interfaces'}->{$interface}->{'rate'} = 100000;
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' has no 'rate' attribute specified, using 100000",$interface); $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' has no 'rate' attribute specified, using 100000",$interface);
} }
# Create interface
my $interfaceID = createInterface({
'ID' => $interface,
'Name' => $interfaceName,
'Device' => $interface,
'Limit' => $interfaceLimit
});
# Check if we have a section in our # Check if we have a section in our
if (defined($iconfig->{'class_rate'})) { if (defined($iconfig->{'class_rate'})) {
...@@ -571,10 +685,10 @@ sub plugin_init ...@@ -571,10 +685,10 @@ sub plugin_init
# Skip comments # Skip comments
next if ($iclass =~ /^\s*#/); next if ($iclass =~ /^\s*#/);
# Split off class ID and class name # Split off class ID and class name
my ($iclassID,$iclassCIR,$iclassLimit) = split(/[:\/]/,$iclass); my ($itrafficClassID,$iclassCIR,$iclassLimit) = split(/[:\/]/,$iclass);
if (!defined(isNumber($iclassID))) { if (!defined($itrafficClassID = isTrafficClassIDValid($itrafficClassID))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class definition '%s' has invalid Class ID, ignoring ". $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class definition '%s' has invalid Class ID, ignoring ".
"definition", "definition",
$interface, $interface,
...@@ -582,15 +696,6 @@ sub plugin_init ...@@ -582,15 +696,6 @@ sub plugin_init
); );
next; next;
} }
if (!defined($config->{'classes'}->{$iclassID})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class definition '%s' uses Class ID '%s' which doesn't ".
"exist",
$interface,
$iclass,
$iclassID
);
next;
}
# If the CIR is defined, try use it # If the CIR is defined, try use it
if (defined($iclassCIR)) { if (defined($iclassCIR)) {
...@@ -599,21 +704,21 @@ sub plugin_init ...@@ -599,21 +704,21 @@ sub plugin_init
my ($cir,$percent) = ($1,$2); my ($cir,$percent) = ($1,$2);
# Check if this is a percentage or an actual kbps value # Check if this is a percentage or an actual kbps value
if (defined($percent)) { if (defined($percent)) {
$iclassCIR = int($config->{'interfaces'}->{$interface}->{'rate'} * ($cir / 100)); $iclassCIR = int($interfaceLimit * ($cir / 100));
} else { } else {
$iclassCIR = $cir; $iclassCIR = $cir;
} }
} else { } else {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has invalid CIR, ignoring definition", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has invalid CIR, ignoring definition",
$interface, $interface,
$iclassID $itrafficClassID
); );
next; next;
} }
} else { } else {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has missing CIR, ignoring definition", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has missing CIR, ignoring definition",
$interface, $interface,
$iclassID $itrafficClassID
); );
next; next;
} }
...@@ -625,54 +730,54 @@ sub plugin_init ...@@ -625,54 +730,54 @@ sub plugin_init
my ($Limit,$percent) = ($1,$2); my ($Limit,$percent) = ($1,$2);
# Check if this is a percentage or an actual kbps value # Check if this is a percentage or an actual kbps value
if (defined($percent)) { if (defined($percent)) {
$iclassLimit = int($config->{'interfaces'}->{$interface}->{'rate'} * ($Limit / 100)); $iclassLimit = int($interfaceLimit * ($Limit / 100));
} else { } else {
$iclassLimit = $Limit; $iclassLimit = $Limit;
} }
} else { } else {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has invalid Limit, ignoring", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has invalid Limit, ignoring",
$interface, $interface,
$iclassID $itrafficClassID
); );
next; next;
} }
} else { } else {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' class '%s' has missing Limit, using CIR '%s' instead", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' class '%s' has missing Limit, using CIR '%s' instead",
$interface, $interface,
$iclassID, $itrafficClassID,
$iclassCIR $iclassCIR
); );
$iclassLimit = $iclassCIR; $iclassLimit = $iclassCIR;
} }
# Check if rates are below are sane # Check if rates are below are sane
if ($iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'}) { if ($iclassCIR > $interfaceLimit) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has CIR '%s' > interface speed '%s', ". $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' class '%s' has CIR '%s' > interface speed '%s', ".
"adjusting to '%s'", "adjusting to '%s'",
$interface, $interface,
$iclassID, $itrafficClassID,
$iclassCIR, $iclassCIR,
$iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'}, $interfaceLimit,
$iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'} $interfaceLimit
); );
$iclassCIR = $iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'}; $iclassCIR = $interfaceLimit;
} }
if ($iclassLimit > $config->{'interfaces'}->{$interface}->{'rate'}) { if ($iclassLimit > $interfaceLimit) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has Limit '%s' > interface speed '%s', ". $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' class '%s' has Limit '%s' > interface speed '%s', ".
"adjusting to '%s'", "adjusting to '%s'",
$interface, $interface,
$iclassID, $itrafficClassID,
$iclassCIR, $iclassCIR,
$iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'}, $interfaceLimit,
$iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'} $interfaceLimit
); );
$iclassLimit = $iclassCIR > $config->{'interfaces'}->{$interface}->{'rate'}; $iclassLimit = $interfaceLimit;
} }
if ($iclassCIR > $iclassLimit) { if ($iclassCIR > $iclassLimit) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' class '%s' has CIR '%s' > Limit '%s', adjusting CIR ". $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' class '%s' has CIR '%s' > Limit '%s', adjusting CIR ".
"to '%s'", "to '%s'",
$interface, $interface,
$iclassID, $itrafficClassID,
$iclassLimit, $iclassLimit,
$iclassLimit, $iclassLimit,
$iclassLimit $iclassLimit
...@@ -680,70 +785,86 @@ sub plugin_init ...@@ -680,70 +785,86 @@ sub plugin_init
$iclassCIR = $iclassLimit; $iclassCIR = $iclassLimit;
} }
# Build class config # Create class
$config->{'interfaces'}->{$interface}->{'classes'}->{$iclassID} = { my $interfaceTrafficClassID = createInterfaceTrafficClass({
'cir' => $iclassCIR, 'InterfaceID' => $interfaceID,
'limit' => $iclassLimit 'TrafficClassID' => $itrafficClassID,
}; 'CIR' => $iclassCIR,
'Limit' => $iclassLimit
});
if (!defined($interfaceTrafficClassID)) {
next;
}
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded interface '%s' class rate for class ID '%s': %s/%s", $logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded interface '%s' class rate for class ID '%s': %s/%s",
$interface, $interface,
$iclassID, $itrafficClassID,
$iclassCIR, $iclassCIR,
$iclassLimit $iclassLimit
); );
} }
# Time to check the interface classes
foreach my $classID (keys %{$config->{'classes'}}) {
# Check if we have a rate defined for this class in the interface definition
if (!defined($config->{'interfaces'}->{$interface}->{'classes'}->{$classID})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface '%s' has no class '%s' defined, using interface limit",
$interface,
$classID
);
$config->{'interfaces'}->{$interface}->{'classes'}->{$classID} = {
'cir' => $config->{'interfaces'}->{$interface}->{'rate'},
'limit' => $config->{'interfaces'}->{$interface}->{'rate'}
};
}
}
} }
# Time to check the interface classes
foreach my $trafficClassID (getAllTrafficClasses()) {
# Check if we have a rate defined for this class in the interface definition
if (!isInterfaceTrafficClassValid($interfaceID,$trafficClassID)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface '%s' has no class '%s' defined, using interface limit",
$interface,
$trafficClassID
);
# Create the default class
createInterfaceTrafficClass({
'InterfaceID' => $interfaceID,
'TrafficClassID' => $trafficClassID,
'CIR' => $interfaceLimit,
'Limit' => $interfaceLimit
});
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded interface '%s' default class rate for class ID '%s': %s/%s",
$interface,
$trafficClassID,
$interfaceLimit,
$interfaceLimit
);
}
}
} }
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interfaces completed"); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interfaces completed");
# Pull in interface groupings # Pull in interface groupings
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interface groups..."); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Loading interface groups...");
# Check if we loaded an array or just text # Check if we loaded an array or just text
my @interfaceGroups; my @cinterfaceGroups;
if (defined($gconfig->{'interface_group'})) { if (defined($gconfig->{'interface_group'})) {
if (ref($gconfig->{'interface_group'}) eq "ARRAY") { if (ref($gconfig->{'interface_group'}) eq "ARRAY") {
@interfaceGroups = @{$gconfig->{'interface_group'}}; @cinterfaceGroups = @{$gconfig->{'interface_group'}};
} else { } else {
@interfaceGroups = ( $gconfig->{'interface_group'} ); @cinterfaceGroups = ( $gconfig->{'interface_group'} );
} }
} else { } else {
@interfaceGroups = ( "eth1,eth0:Default" ); @cinterfaceGroups = ( "eth1,eth0:Default" );
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No interface groups, trying default eth1,eth0"); $logger->log(LOG_NOTICE,"[CONFIGMANAGER] No interface groups, trying default eth1,eth0");
} }
# Loop with interface groups # Loop with interface groups
foreach my $interfaceGroup (@interfaceGroups) { foreach my $interfaceGroup (@cinterfaceGroups) {
# Skip comments # Skip comments
next if ($interfaceGroup =~ /^\s*#/); next if ($interfaceGroup =~ /^\s*#/);
# Split off class ID and class name # Split off class ID and class name
my ($txiface,$rxiface,$friendlyName) = split(/[:,]/,$interfaceGroup); my ($txInterface,$rxInterface,$friendlyName) = split(/[:,]/,$interfaceGroup);
if (!defined($config->{'interfaces'}->{$txiface})) { if (!isInterfaceIDValid($txInterface)) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface group definition '%s' has invalid interface '%s', ignoring", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface group definition '%s' has invalid interface '%s', ignoring",
$interfaceGroup, $interfaceGroup,
$txiface $txInterface
); );
next; next;
} }
if (!defined($config->{'interfaces'}->{$rxiface})) { if (!isInterfaceIDValid($rxInterface)) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Interface group definition '%s' has invalid interface '%s', ignoring", $logger->log(LOG_WARN,"[CONFIGMANAGER] Interface group definition '%s' has invalid interface '%s', ignoring",
$interfaceGroup, $interfaceGroup,
$rxiface $rxInterface
); );
next; next;
} }
...@@ -754,25 +875,25 @@ sub plugin_init ...@@ -754,25 +875,25 @@ sub plugin_init
next; next;
} }
$config->{'interface_groups'}->{"$txiface,$rxiface"} = { # Create interface group
'name' => $friendlyName, my $interfaceGroupID = createInterfaceGroup({
'txiface' => $txiface, 'Name' => $friendlyName,
'rxiface' => $rxiface 'TxInterface' => $txInterface,
}; 'RxInterface' => $rxInterface
});
if (!defined($interfaceGroupID)) {
next;
}
$logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded interface group '%s' with interfaces '%s/%s'", $logger->log(LOG_INFO,"[CONFIGMANAGER] Loaded interface group '%s' [%s] with interfaces '%s/%s'",
$friendlyName, $friendlyName,
$txiface, $interfaceGroupID,
$rxiface $txInterface,
$rxInterface
); );
} }
# Initialize IP address map
foreach my $interfaceGroupID (keys %{$config->{'interface_groups'}}) {
# Blank interface IP address map for interface group
$interfaceIPMap->{$interfaceGroupID} = { };
}
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Interface groups loaded"); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Interface groups loaded");
...@@ -780,12 +901,11 @@ sub plugin_init ...@@ -780,12 +901,11 @@ sub plugin_init
if (defined($gconfig->{'default_pool'})) { if (defined($gconfig->{'default_pool'})) {
# Check if its a number # Check if its a number
if (defined(my $default_pool = isNumber($gconfig->{'default_pool'}))) { if (defined(my $default_pool = isNumber($gconfig->{'default_pool'}))) {
if (defined($config->{'classes'}->{$default_pool})) { if (isTrafficClassIDValid($default_pool)) {
$logger->log(LOG_INFO,"[CONFIGMANAGER] Default pool set to use class '%s' (%s)", $logger->log(LOG_INFO,"[CONFIGMANAGER] Default pool set to use class '%s'",
$default_pool, $default_pool
$config->{'classes'}->{$default_pool}
); );
$config->{'default_pool'} = $default_pool; $globals->{'DefaultPool'} = $default_pool;
} else { } else {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot enable default pool, class '%s' does not exist", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot enable default pool, class '%s' does not exist",
$default_pool $default_pool
...@@ -797,7 +917,7 @@ sub plugin_init ...@@ -797,7 +917,7 @@ sub plugin_init
} }
# Check if we have a state file # Check if we have a state file
if (defined(my $statefile = $globals->{'file.config'}->{'system'}->{'statefile'})) { if (defined(my $statefile = $system->{'file.config'}->{'system'}->{'statefile'})) {
$config->{'statefile'} = $statefile; $config->{'statefile'} = $statefile;
$logger->log(LOG_INFO,"[CONFIGMANAGER] Set statefile to '%s'",$statefile); $logger->log(LOG_INFO,"[CONFIGMANAGER] Set statefile to '%s'",$statefile);
} }
...@@ -812,9 +932,9 @@ sub plugin_init ...@@ -812,9 +932,9 @@ sub plugin_init
limit_add => \&_session_limit_add, limit_add => \&_session_limit_add,
override_add => \&_session_override_add, pool_override_add => \&_session_pool_override_add,
override_change => \&_session_override_change, pool_override_change => \&_session_pool_override_change,
override_remove => \&_session_override_remove, pool_override_remove => \&_session_pool_override_remove,
pool_add => \&_session_pool_add, pool_add => \&_session_pool_add,
pool_remove => \&_session_pool_remove, pool_remove => \&_session_pool_remove,
...@@ -829,6 +949,7 @@ sub plugin_init ...@@ -829,6 +949,7 @@ sub plugin_init
} }
# Start the plugin # Start the plugin
sub plugin_start sub plugin_start
{ {
...@@ -839,10 +960,10 @@ sub plugin_start ...@@ -839,10 +960,10 @@ sub plugin_start
$logger->log(LOG_WARN,"[CONFIGMANAGER] Statefile '%s' cannot be opened: %s",$config->{'statefile'},$!); $logger->log(LOG_WARN,"[CONFIGMANAGER] Statefile '%s' cannot be opened: %s",$config->{'statefile'},$!);
} }
$logger->log(LOG_INFO,"[CONFIGMANAGER] Started with %s pools, %s pool members and %s overrides", $logger->log(LOG_INFO,"[CONFIGMANAGER] Started with %s pools, %s pool members and %s pool overrides",
scalar(keys %{$pools}), scalar(keys %{$globals->{'Pools'}}),
scalar(keys %{$poolMembers}), scalar(keys %{$globals->{'PoolMembers'}}),
scalar(keys %{$overrides}) scalar(keys %{$globals->{'PoolOverrides'}})
); );
} }
...@@ -866,6 +987,7 @@ sub _session_start ...@@ -866,6 +987,7 @@ sub _session_start
} }
# Stop the session # Stop the session
sub _session_stop sub _session_stop
{ {
...@@ -875,7 +997,7 @@ sub _session_stop ...@@ -875,7 +997,7 @@ sub _session_stop
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Shutting down, saving configuration..."); $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Shutting down, saving configuration...");
# We only need to write the sate if something changed? # We only need to write the sate if something changed?
if ($stateChanged) { if ($globals->{'StateChanged'}) {
# The 1 means FULL WRITE of all entries # The 1 means FULL WRITE of all entries
_write_statefile(1); _write_statefile(1);
} }
...@@ -883,27 +1005,13 @@ sub _session_stop ...@@ -883,27 +1005,13 @@ sub _session_stop
# Blow away all data # Blow away all data
$globals = undef; $globals = undef;
$interfaceIPMap = { };
$pools = { };
$poolNameMap = { };
$poolIDCounter = 1;
$poolMembers = { };
$poolMemberIDCounter = 1;
$poolMemberMap = { };
$poolChangeQueue = { };
$poolMemberChangeQueue = { };
# XXX: Blow away rest? config?
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Shutdown"); $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Shutdown");
$logger = undef; $logger = undef;
} }
# Time ticker for processing changes # Time ticker for processing changes
sub _session_tick sub _session_tick
{ {
...@@ -913,25 +1021,26 @@ sub _session_tick ...@@ -913,25 +1021,26 @@ sub _session_tick
my $now = time(); my $now = time();
# Check if we should sync state to disk # Check if we should sync state to disk
if ($stateChanged && $lastStateSync + STATE_SYNC_INTERVAL < $now) { if ($globals->{'StateChanged'} && $globals->{'LastStateSync'} + STATE_SYNC_INTERVAL < $now) {
_write_statefile(); _write_statefile();
} }
# Check if we should cleanup # Check if we should cleanup
if ($lastCleanup + CLEANUP_INTERVAL < $now) { if ($globals->{'LastCleanup'} + CLEANUP_INTERVAL < $now) {
# Loop with all overrides and check for expired entries # Loop with all pool overrides and check for expired entries
while (my ($oid, $override) = each(%{$overrides})) { while (my ($poid, $poolOverride) = each(%{$globals->{'PoolOverrides'}})) {
# Override has effectively expired # Pool override has effectively expired
if (defined($override->{'Expires'}) && $override->{'Expires'} > 0 && $override->{'Expires'} < $now) { if (defined($poolOverride->{'Expires'}) && $poolOverride->{'Expires'} > 0 && $poolOverride->{'Expires'} < $now) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Override '%s' [%s] has expired, removing", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool override '%s' [%s] has expired, removing",
$override->{'FriendlyName'}, $poolOverride->{'FriendlyName'},
$oid $poid
); );
removeOverride($oid); removePoolOverride($poid);
} }
} }
# Loop with all pool members and check for expired entries # Loop with all pool members and check for expired entries
while (my ($pmid, $poolMember) = each(%{$poolMembers})) { while (my ($pmid, $poolMember) = each(%{$globals->{'PoolMembers'}})) {
# Pool member has effectively expired # Pool member has effectively expired
if (defined($poolMember->{'Expires'}) && $poolMember->{'Expires'} > 0 && $poolMember->{'Expires'} < $now) { if (defined($poolMember->{'Expires'}) && $poolMember->{'Expires'} > 0 && $poolMember->{'Expires'} < $now) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool member '%s' [%s] has expired, removing", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool member '%s' [%s] has expired, removing",
...@@ -942,7 +1051,7 @@ sub _session_tick ...@@ -942,7 +1051,7 @@ sub _session_tick
} }
} }
# Loop with all the pools and check for expired entries # Loop with all the pools and check for expired entries
while (my ($pid, $pool) = each(%{$pools})) { while (my ($pid, $pool) = each(%{$globals->{'Pools'}})) {
# Pool has effectively expired # Pool has effectively expired
if (defined($pool->{'Expires'}) && $pool->{'Expires'} > 0 && $pool->{'Expires'} < $now) { if (defined($pool->{'Expires'}) && $pool->{'Expires'} > 0 && $pool->{'Expires'} < $now) {
# There are no members, its safe to remove # There are no members, its safe to remove
...@@ -956,12 +1065,46 @@ sub _session_tick ...@@ -956,12 +1065,46 @@ sub _session_tick
} }
} }
# Reset last cleanup time # Reset last cleanup time
$lastCleanup = $now; $globals->{'LastCleanup'} = $now;
} }
# Loop through interface traffic classes
while (my ($interfaceTrafficClassID, $interfaceTrafficClass) = each(%{$globals->{'InterfaceTrafficClassChangeQueue'}})) {
my $shaperState = getInterfaceTrafficClassShaperState($interfaceTrafficClassID);
# Traffic class has been changed
if ($interfaceTrafficClass->{'Status'} == CFGM_CHANGED) {
# If the shaper is live we can go ahead
if ($shaperState & SHAPER_LIVE) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Interface traffic class [%s] has been modified, sending to shaper",
$interfaceTrafficClassID
);
$kernel->post('shaper' => 'class_change' => $interfaceTrafficClassID);
# Set pending online
setInterfaceTrafficClassShaperState($interfaceTrafficClassID,SHAPER_PENDING);
$interfaceTrafficClass->{'Status'} = CFGM_ONLINE;
# Remove from queue
delete($globals->{'InterfaceTrafficClassChangeQueue'}->{$interfaceTrafficClassID});
} else {
$logger->log(LOG_ERR,"[CONFIGMANAGER] Interface traffic class [%s] has UNKNOWN state '%s'",
$interfaceTrafficClassID,
$shaperState
);
}
} else {
$logger->log(LOG_ERR,"[CONFIGMANAGER] Interface traffic class [%s] has UNKNOWN status '%s'",
$interfaceTrafficClassID,
$interfaceTrafficClass->{'Status'}
);
}
}
# Loop through pool change queue # Loop through pool change queue
while (my ($pid, $pool) = each(%{$poolChangeQueue})) { while (my ($pid, $pool) = each(%{$globals->{'PoolChangeQueue'}})) {
my $shaperState = getPoolShaperState($pool->{'ID'}); my $shaperState = getPoolShaperState($pid);
# Pool is newly added # Pool is newly added
if ($pool->{'Status'} == CFGM_NEW) { if ($pool->{'Status'} == CFGM_NEW) {
...@@ -974,10 +1117,10 @@ sub _session_tick ...@@ -974,10 +1117,10 @@ sub _session_tick
); );
$kernel->post('shaper' => 'pool_add' => $pid); $kernel->post('shaper' => 'pool_add' => $pid);
# Set pending online # Set pending online
setPoolShaperState($pool->{'ID'},SHAPER_PENDING); setPoolShaperState($pid,SHAPER_PENDING);
$pool->{'Status'} = CFGM_ONLINE; $pool->{'Status'} = CFGM_ONLINE;
# Remove from queue # Remove from queue
delete($poolChangeQueue->{$pid}); delete($globals->{'PoolChangeQueue'}->{$pid});
} else { } else {
$logger->log(LOG_ERR,"[CONFIGMANAGER] Pool '%s' [%s] has UNKNOWN state '%s'", $logger->log(LOG_ERR,"[CONFIGMANAGER] Pool '%s' [%s] has UNKNOWN state '%s'",
...@@ -1010,7 +1153,6 @@ sub _session_tick ...@@ -1010,7 +1153,6 @@ sub _session_tick
# Pool has been modified # Pool has been modified
} elsif ($pool->{'Status'} == CFGM_CHANGED) { } elsif ($pool->{'Status'} == CFGM_CHANGED) {
# If the shaper is live we can go ahead # If the shaper is live we can go ahead
if ($shaperState & SHAPER_LIVE) { if ($shaperState & SHAPER_LIVE) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] has been modified, sending to shaper", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] has been modified, sending to shaper",
...@@ -1019,10 +1161,10 @@ sub _session_tick ...@@ -1019,10 +1161,10 @@ sub _session_tick
); );
$kernel->post('shaper' => 'pool_change' => $pid); $kernel->post('shaper' => 'pool_change' => $pid);
# Set pending online # Set pending online
setPoolShaperState($pool->{'ID'},SHAPER_PENDING); setPoolShaperState($pid,SHAPER_PENDING);
$pool->{'Status'} = CFGM_ONLINE; $pool->{'Status'} = CFGM_ONLINE;
# Remove from queue # Remove from queue
delete($poolChangeQueue->{$pid}); delete($globals->{'PoolChangeQueue'}->{$pid});
} elsif ($shaperState & SHAPER_NOTLIVE) { } elsif ($shaperState & SHAPER_NOTLIVE) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] has been modified and is not live, re-queue as add", $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] has been modified and is not live, re-queue as add",
...@@ -1054,7 +1196,7 @@ sub _session_tick ...@@ -1054,7 +1196,7 @@ sub _session_tick
$pid $pid
); );
$kernel->post('shaper' => 'pool_remove' => $pid); $kernel->post('shaper' => 'pool_remove' => $pid);
setPoolShaperState($pool->{'ID'},SHAPER_PENDING); setPoolShaperState($pid,SHAPER_PENDING);
} else { } else {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] marked offline, but still has pool members, ". $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' [%s] marked offline, but still has pool members, ".
"aborting remove", "aborting remove",
...@@ -1062,7 +1204,7 @@ sub _session_tick ...@@ -1062,7 +1204,7 @@ sub _session_tick
$pid $pid
); );
$pool->{'Status'} = CFGM_ONLINE; $pool->{'Status'} = CFGM_ONLINE;
delete($poolChangeQueue->{$pid}); delete($globals->{'PoolChangeQueue'}->{$pid});
} }
} else { } else {
...@@ -1070,7 +1212,7 @@ sub _session_tick ...@@ -1070,7 +1212,7 @@ sub _session_tick
if (my @poolMembers = getPoolMembers($pid)) { if (my @poolMembers = getPoolMembers($pid)) {
# Loop with members and remove # Loop with members and remove
foreach my $pmid (@poolMembers) { foreach my $pmid (@poolMembers) {
my $poolMember = $poolMembers->{$pmid}; my $poolMember = $globals->{'PoolMembers'}->{$pmid};
# Only remove ones online # Only remove ones online
if ($poolMember->{'Status'} == CFGM_ONLINE) { if ($poolMember->{'Status'} == CFGM_ONLINE) {
$logger->log(LOG_INFO,"[CONFIGMANAGER] Pool '%s' [%s] marked offline and not expired, removing ". $logger->log(LOG_INFO,"[CONFIGMANAGER] Pool '%s' [%s] marked offline and not expired, removing ".
...@@ -1091,15 +1233,15 @@ sub _session_tick ...@@ -1091,15 +1233,15 @@ sub _session_tick
$pid $pid
); );
# Remove pool from name map # Remove pool from name map
delete($poolNameMap->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}}); delete($globals->{'PoolNameMap'}->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}});
# Remove pool member mapping # Remove pool member mapping
delete($poolMemberMap->{$pool->{'ID'}}); delete($globals->{'PoolMemberMap'}->{$pid});
# Remove from queue # Remove from queue
delete($poolChangeQueue->{$pid}); delete($globals->{'PoolChangeQueue'}->{$pid});
# Cleanup overrides # Cleanup pool overrides
_override_remove_pool($pool->{'ID'}); _remove_pool_override($pid);
# Remove pool # Remove pool
delete($pools->{$pool->{'ID'}}); delete($globals->{'Pools'}->{$pid});
} }
} else { } else {
...@@ -1111,17 +1253,18 @@ sub _session_tick ...@@ -1111,17 +1253,18 @@ sub _session_tick
} }
} }
# Loop through pool member change queue # Loop through pool member change queue
while (my ($pmid, $poolMember) = each(%{$poolMemberChangeQueue})) { while (my ($pmid, $poolMember) = each(%{$globals->{'PoolMemberChangeQueue'}})) {
my $pool = $pools->{$poolMember->{'PoolID'}}; my $pool = $globals->{'Pools'}->{$poolMember->{'PoolID'}};
# We need to skip doing anything until the pool becomes live # We need to skip doing anything until the pool becomes live
if (getPoolShaperState($pool->{'ID'}) & SHAPER_NOTLIVE) { if (getPoolShaperState($pool->{'ID'}) & SHAPER_NOTLIVE) {
next; next;
} }
my $shaperState = getPoolMemberShaperState($poolMember->{'ID'}); my $shaperState = getPoolMemberShaperState($pmid);
# Pool member is newly added # Pool member is newly added
if ($poolMember->{'Status'} == CFGM_NEW) { if ($poolMember->{'Status'} == CFGM_NEW) {
...@@ -1135,10 +1278,10 @@ sub _session_tick ...@@ -1135,10 +1278,10 @@ sub _session_tick
); );
$kernel->post('shaper' => 'poolmember_add' => $pmid); $kernel->post('shaper' => 'poolmember_add' => $pmid);
# Set pending online # Set pending online
setPoolMemberShaperState($poolMember->{'ID'},SHAPER_PENDING); setPoolMemberShaperState($pmid,SHAPER_PENDING);
$poolMember->{'Status'} = CFGM_ONLINE; $poolMember->{'Status'} = CFGM_ONLINE;
# Remove from queue # Remove from queue
delete($poolMemberChangeQueue->{$pmid}); delete($globals->{'PoolMemberChangeQueue'}->{$pmid});
} else { } else {
$logger->log(LOG_ERR,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] has UNKNOWN state '%s'", $logger->log(LOG_ERR,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] has UNKNOWN state '%s'",
...@@ -1184,10 +1327,10 @@ sub _session_tick ...@@ -1184,10 +1327,10 @@ sub _session_tick
); );
$kernel->post('shaper' => 'poolmember_change' => $pmid); $kernel->post('shaper' => 'poolmember_change' => $pmid);
# Set pending online # Set pending online
setPoolMemberShaperState($poolMember->{'ID'},SHAPER_PENDING); setPoolMemberShaperState($pmid,SHAPER_PENDING);
$poolMember->{'Status'} = CFGM_ONLINE; $poolMember->{'Status'} = CFGM_ONLINE;
# Remove from queue # Remove from queue
delete($poolMemberChangeQueue->{$pmid}); delete($globals->{'PoolMemberChangeQueue'}->{$pmid});
} elsif ($shaperState & SHAPER_NOTLIVE) { } elsif ($shaperState & SHAPER_NOTLIVE) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] has been modified and is not live, re-queue as ". $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] has been modified and is not live, re-queue as ".
...@@ -1222,7 +1365,7 @@ sub _session_tick ...@@ -1222,7 +1365,7 @@ sub _session_tick
$pmid $pmid
); );
$kernel->post('shaper' => 'poolmember_remove' => $pmid); $kernel->post('shaper' => 'poolmember_remove' => $pmid);
setPoolMemberShaperState($poolMember->{'ID'},SHAPER_PENDING); setPoolMemberShaperState($pmid,SHAPER_PENDING);
} else { } else {
$logger->log(LOG_INFO,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] marked offline and fresh, postponing", $logger->log(LOG_INFO,"[CONFIGMANAGER] Pool '%s' member '%s' [%s] marked offline and fresh, postponing",
...@@ -1239,15 +1382,63 @@ sub _session_tick ...@@ -1239,15 +1382,63 @@ sub _session_tick
$pmid $pmid
); );
# Unlink interface IP address map # Unlink interface IP address map
delete($interfaceIPMap->{$pool->{'InterfaceGroupID'}}->{$poolMember->{'IPAddress'}}->{$poolMember->{'ID'}}); delete($globals->{'InterfaceGroups'}->{$pool->{'InterfaceGroupID'}}->{'IPMap'}->{$poolMember->{'IPAddress'}}
->{$pmid});
# Unlink pool map # Unlink pool map
delete($poolMemberMap->{$pool->{'ID'}}->{$poolMember->{'ID'}}); delete($globals->{'PoolMemberMap'}->{$pool->{'ID'}}->{$pmid});
# Remove from queue # Remove from queue
delete($poolMemberChangeQueue->{$pmid}); delete($globals->{'PoolMemberChangeQueue'}->{$pmid});
# We need to re-process the overrides after the member has been removed # We need to re-process the pool overrides after the member has been removed
_override_resolve([$poolMember->{'PoolID'}]); _resolve_pool_override([$poolMember->{'PoolID'}]);
# Remove pool member # Remove pool member
delete($poolMembers->{$poolMember->{'ID'}}); delete($globals->{'PoolMembers'}->{$pmid});
# Check if we have/had conflicts
if ((my @conflicts = keys
%{$globals->{'InterfaceGroups'}->{$pool->{'InterfaceGroupID'}}->{'IPMap'}->{$poolMember->{'IPAddress'}}}) > 0)
{
# We can only re-tag a pool member for adding if we have 1 pool member
if (@conflicts == 1) {
# Grab conflicted pool member, its index 0 in the conflicts array
my $cPoolMember = $globals->{'PoolMembers'}->{$conflicts[0]};
my $cPoolMemberShaperState = getPoolMemberShaperState($cPoolMember->{'ID'});
# We only want to work with conflicts
if ($cPoolMemberShaperState & SHAPER_CONFLICT) {
# Grab pool
my $cPool = $globals->{'Pools'}->{$cPoolMember->{'PoolID'}};
# Unset conflict state
unsetPoolMemberShaperState($cPoolMember->{'ID'},SHAPER_CONFLICT);
# Add to change queue
$globals->{'PoolMemberChangeQueue'}->{$cPoolMember->{'ID'}} = $cPoolMember;
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] IP '%s' is no longer conflicted, removing conflict from ".
"pool '%s' member '%s' [%s], was conflicted with pool '%s' member '%s' [%s]",
$cPoolMember->{'IPAddress'},
$cPool->{'Name'},
$cPoolMember->{'Username'},
$cPoolMember->{'ID'},
$pool->{'Name'},
$poolMember->{'Username'},
$poolMember->{'ID'}
);
}
} else {
# Loop wiht conflicts and build some log items to use
my @logItems;
foreach my $pmid (@conflicts) {
my $cPoolMember = $globals->{'PoolMembers'}->{$pmid};
my $cPool = $globals->{'Pools'}->{$cPoolMember->{'PoolID'}};
push(@logItems,sprintf("Pool:%s/Member:%s",$cPool->{'Name'},$cPoolMember->{'Username'}));
}
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] IP '%s' is still in conflict: %s",
$poolMember->{'IPAddress'},
join(", ",@logItems)
);
}
}
} }
} else { } else {
...@@ -1266,15 +1457,17 @@ sub _session_tick ...@@ -1266,15 +1457,17 @@ sub _session_tick
} }
# Handle SIGHUP # Handle SIGHUP
sub _session_SIGHUP sub _session_SIGHUP
{ {
my ($kernel, $heap, $signal_name) = @_[KERNEL, HEAP, ARG0]; my ($kernel, $heap, $signal_name) = @_[KERNEL, HEAP, ARG0];
$logger->log(LOG_WARN,"[CONFIGMANAGER] Got SIGHUP, ignoring for now"); $logger->log(LOG_NOTICE,"[CONFIGMANAGER] Got SIGHUP, ignoring for now");
} }
# Event for 'pool_add' # Event for 'pool_add'
sub _session_pool_add sub _session_pool_add
{ {
...@@ -1282,7 +1475,7 @@ sub _session_pool_add ...@@ -1282,7 +1475,7 @@ sub _session_pool_add
if (!defined($poolData)) { if (!defined($poolData)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No pool data provided for 'pool_add' event"); $logger->log(LOG_WARN,"[CONFIGMANAGER] No pool data provided for 'pool_add' event");
return; return;
} }
...@@ -1305,6 +1498,7 @@ sub _session_pool_add ...@@ -1305,6 +1498,7 @@ sub _session_pool_add
} }
# Event for 'pool_remove' # Event for 'pool_remove'
sub _session_pool_remove sub _session_pool_remove
{ {
...@@ -1313,7 +1507,7 @@ sub _session_pool_remove ...@@ -1313,7 +1507,7 @@ sub _session_pool_remove
my $pool; my $pool;
if (!defined(getPool($pid))) { if (!defined(getPool($pid))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid pool ID '%s' for 'pool_remove' event",prettyUndef($pid)); $logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool ID '%s' for 'pool_remove' event",prettyUndef($pid));
return; return;
} }
...@@ -1321,6 +1515,7 @@ sub _session_pool_remove ...@@ -1321,6 +1515,7 @@ sub _session_pool_remove
} }
# Event for 'pool_change' # Event for 'pool_change'
sub _session_pool_change sub _session_pool_change
{ {
...@@ -1328,7 +1523,7 @@ sub _session_pool_change ...@@ -1328,7 +1523,7 @@ sub _session_pool_change
if (!isPoolIDValid($poolData->{'ID'})) { if (!isPoolIDValid($poolData->{'ID'})) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid pool ID '%s' for 'pool_change' event",prettyUndef($poolData->{'ID'})); $logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool ID '%s' for 'pool_change' event",prettyUndef($poolData->{'ID'}));
return; return;
} }
...@@ -1336,6 +1531,7 @@ sub _session_pool_change ...@@ -1336,6 +1531,7 @@ sub _session_pool_change
} }
# Event for 'poolmember_add' # Event for 'poolmember_add'
sub _session_poolmember_add sub _session_poolmember_add
{ {
...@@ -1343,7 +1539,7 @@ sub _session_poolmember_add ...@@ -1343,7 +1539,7 @@ sub _session_poolmember_add
if (!defined($poolMemberData)) { if (!defined($poolMemberData)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No pool member data provided for 'poolmember_add' event"); $logger->log(LOG_WARN,"[CONFIGMANAGER] No pool member data provided for 'poolmember_add' event");
return; return;
} }
...@@ -1364,192 +1560,666 @@ sub _session_poolmember_add ...@@ -1364,192 +1560,666 @@ sub _session_poolmember_add
} }
# Event for 'poolmember_remove' # Event for 'poolmember_remove'
sub _session_poolmember_remove sub _session_poolmember_remove
{ {
my ($kernel, $pmid) = @_[KERNEL, ARG0]; my ($kernel, $pmid) = @_[KERNEL, ARG0];
if (!isPoolMemberIDValid($pmid)) { if (!isPoolMemberIDValid($pmid)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid pool member ID '%s' for 'poolmember_remove' event",prettyUndef($pmid)); $logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool member ID '%s' for 'poolmember_remove' event",prettyUndef($pmid));
return;
}
removePoolMember($pmid);
}
# Event for 'poolmember_change'
sub _session_poolmember_change
{
my ($kernel, $poolMemberData) = @_[KERNEL, ARG0];
if (!isPoolMemberIDValid($poolMemberData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool member ID '%s' for 'poolmember_change' event",
prettyUndef($poolMemberData->{'ID'})
);
return;
}
changePoolMember($poolMemberData);
}
# Event for 'limit_add'
sub _session_limit_add
{
my ($kernel, $limitData) = @_[KERNEL, ARG0];
if (!defined($limitData)) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] No limit data provided for 'limit_add' event");
return;
}
# Check if we have all the attributes we need
my $isInvalid;
foreach my $attr (LIMIT_REQUIRED_ATTRIBUTES) {
if (!defined($limitData->{$attr})) {
$isInvalid = $attr;
last;
}
}
if ($isInvalid) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process limit add as there is an attribute missing: '%s'",$isInvalid);
return;
}
createLimit($limitData);
}
# Event for 'pool_override_add'
sub _session_pool_override_add
{
my ($kernel, $poolOverrideData) = @_[KERNEL, ARG0];
if (!defined($poolOverrideData)) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] No pool override data provided for 'pool_override_add' event");
return;
}
# Check that we have at least one match attribute
my $isValid = 0;
foreach my $item (POOL_OVERRIDE_MATCH_ATTRIBUTES) {
$isValid++;
}
if (!$isValid) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool override as there is no selection attribute");
return;
}
createPoolOverride($poolOverrideData);
}
# Event for 'pool_override_remove'
sub _session_pool_override_remove
{
my ($kernel, $poid) = @_[KERNEL, ARG0];
if (!isPoolOverrideIDValid($poid)) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool override ID '%s' for 'pool_override_remove' event",
prettyUndef($poid)
);
return;
}
removePoolOverride($poid);
}
# Event for 'pool_override_change'
sub _session_pool_override_change
{
my ($kernel, $poolOverrideData) = @_[KERNEL, ARG0];
if (!isPoolOverrideIDValid($poolOverrideData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Invalid pool override ID '%s' for 'pool_override_change' event",
prettyUndef($poolOverrideData->{'ID'})
);
return;
}
changePoolOverride($poolOverrideData);
}
# Function to create a group
sub createGroup
{
my $groupData = shift;
my $group;
# Check if ID is valid
if (!defined($group->{'ID'} = $groupData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add group as ID is invalid");
return;
}
# Check if Name is valid
if (!defined($group->{'Name'} = $groupData->{'Name'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add group as Name is invalid");
return;
}
# Add pool
$globals->{'Groups'}->{$group->{'ID'}} = $group;
return $group->{'ID'};
}
# Function to check the group ID exists
sub isGroupIDValid
{
my $gid = shift;
if (!defined($globals->{'Groups'}->{$gid})) {
return;
}
return $gid;
}
# Function to create a traffic class
sub createTrafficClass
{
my $classData = shift;
my $class;
# Check if ID is valid
if (!defined($class->{'ID'} = $classData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add traffic class as ID is invalid");
return;
}
# Check if Name is valid
if (!defined($class->{'Name'} = $classData->{'Name'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add traffic class as Name is invalid");
return;
}
# Add pool
$globals->{'TrafficClasses'}->{$class->{'ID'}} = $class;
return $class->{'ID'};
}
# Function to get traffic classes
sub getTrafficClasses
{
my @trafficClasses = ( );
# Loop with traffic classes
foreach my $trafficClassID (keys %{$globals->{'TrafficClasses'}}) {
# Skip over default pool if we have one
if (defined($globals->{'DefaultPool'}) && $trafficClassID eq $globals->{'DefaultPool'}) {
next;
}
# Add to class list
push (@trafficClasses,$trafficClassID);
}
return @trafficClasses;
}
# Function to get a interface traffic class
sub getInterfaceTrafficClass
{
my ($interfaceID,$trafficClassID) = @_;
# Check if this interface ID is valid
if (!isInterfaceIDValid($interfaceID)) {
return;
}
# Check if traffic class ID is valid
if (!defined($trafficClassID = isNumber($trafficClassID,ISNUMBER_ALLOW_ZERO))) {
return;
}
if ($trafficClassID && !isTrafficClassIDValid($trafficClassID)) {
return;
}
my $interfaceTrafficClass = dclone($globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{$trafficClassID});
# Check if the traffic class ID is not 0
if ($trafficClassID) {
$interfaceTrafficClass->{'Name'} = $globals->{'TrafficClasses'}->{$trafficClassID}->{'Name'};
# If if it 0, this is a root class
} else {
$interfaceTrafficClass->{'Name'} = "Root Class";
}
delete($interfaceTrafficClass->{'.applied_overrides'});
return $interfaceTrafficClass;
}
# Function to get a interface traffic class
sub getInterfaceTrafficClass2
{
my $interfaceTrafficClassID = shift;
# Check if this interface ID is valid
if (!isInterfaceTrafficClassIDValid2($interfaceTrafficClassID)) {
return;
}
my $interfaceTrafficClass = dclone($globals->{'InterfaceTrafficClasses'}->{$interfaceTrafficClassID});
$interfaceTrafficClass->{'Name'} = $globals->{'TrafficClasses'}->{$interfaceTrafficClass->{'TrafficClassID'}};
delete($interfaceTrafficClass->{'.applied_overrides'});
return $interfaceTrafficClass;
}
# Function to check if traffic class is valid
sub isInterfaceTrafficClassValid
{
my ($interfaceID,$trafficClassID) = @_;
if (
!defined($interfaceID) || !defined($trafficClassID) ||
!defined($globals->{'Interfaces'}->{$interfaceID}) ||
!defined($globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{$trafficClassID})
) {
return;
}
return $globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{$trafficClassID}->{'ID'};
}
# Function to check the interface traffic class ID is valid
sub isInterfaceTrafficClassIDValid2
{
my $interfaceTrafficClassID = shift;
if (
!defined($interfaceTrafficClassID) ||
!defined($globals->{'InterfaceTrafficClasses'}->{$interfaceTrafficClassID})
) {
return;
}
return $interfaceTrafficClassID;
}
# Function to create an interface class
sub createInterfaceTrafficClass
{
my $interfaceTrafficClassData = shift;
my $interfaceTrafficClass;
# Check if InterfaceID is valid
if (!defined($interfaceTrafficClass->{'InterfaceID'} = isInterfaceIDValid($interfaceTrafficClassData->{'InterfaceID'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface traffic class as InterfaceID is invalid");
return;
}
# Check if traffic class ID is valid
my $interfaceTrafficClassID;
if (!defined($interfaceTrafficClassID = isNumber($interfaceTrafficClassData->{'TrafficClassID'},ISNUMBER_ALLOW_ZERO))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process class change as there is no 'TrafficClassID' attribute");
return;
}
if ($interfaceTrafficClassID && !isTrafficClassIDValid($interfaceTrafficClassData->{'TrafficClassID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process class change as 'TrafficClassID' attribute is invalid");
return;
}
$interfaceTrafficClass->{'TrafficClassID'} = $interfaceTrafficClassID;
# Check CIR is valid
if (!defined($interfaceTrafficClass->{'CIR'} = isNumber($interfaceTrafficClassData->{'CIR'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as CIR is invalid");
return;
}
# Check Limit is valid
if (!defined($interfaceTrafficClass->{'Limit'} = isNumber($interfaceTrafficClassData->{'Limit'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as Limit is invalid");
return;
}
# Set ID
$interfaceTrafficClass->{'ID'} = $globals->{'InterfaceTrafficClassCounter'}++;
# Set status
$interfaceTrafficClass->{'Status'} = CFGM_NEW;
# Add interface
$globals->{'Interfaces'}->{$interfaceTrafficClass->{'InterfaceID'}}->{'TrafficClasses'}
->{$interfaceTrafficClass->{'TrafficClassID'}} = $interfaceTrafficClass;
# Link to interface traffic classes
$globals->{'InterfaceTrafficClasses'}->{$interfaceTrafficClass->{'ID'}} = $interfaceTrafficClass;
# TODO: Hack, this should set NOTLIVE & NEW and have the shaper create as per note in plugin_init section
# Set status on this interface traffic class
setInterfaceTrafficClassShaperState($interfaceTrafficClass->{'ID'},SHAPER_LIVE);
$interfaceTrafficClass->{'Status'} = CFGM_ONLINE;
return $interfaceTrafficClass->{'TrafficClassID'};
}
# Function to change a traffic class
sub changeInterfaceTrafficClass
{
my $interfaceTrafficClassData = shift;
# Check interface exists first
my $interfaceID;
if (!defined($interfaceID = isInterfaceIDValid($interfaceTrafficClassData->{'InterfaceID'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process interface class change as there is no 'InterfaceID' attribute");
return;
}
# Check if traffic class ID is valid
my $trafficClassID;
if (!defined($trafficClassID = isNumber($interfaceTrafficClassData->{'TrafficClassID'},ISNUMBER_ALLOW_ZERO))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process class change as there is no 'TrafficClassID' attribute");
return;
}
if ($trafficClassID && !isTrafficClassIDValid($interfaceTrafficClassData->{'TrafficClassID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process class change as 'TrafficClassID' attribute is invalid");
return;
}
my $interfaceTrafficClass = $globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{$trafficClassID};
my $changes = getHashChanges($interfaceTrafficClass,$interfaceTrafficClassData,[CLASS_CHANGE_ATTRIBUTES]);
# Bump up changes
$globals->{'StateChanged'}++;
# Flag changed
$interfaceTrafficClass->{'Status'} = CFGM_CHANGED;
# XXX - hack our override in
$interfaceTrafficClass->{'.applied_overrides'}->{'change'} = $changes;
# Add to change queue
$globals->{'InterfaceTrafficClassChangeQueue'}->{$interfaceTrafficClass->{'ID'}} = $interfaceTrafficClass;
# Return what was changed
return dclone($changes);
}
# Function to return a class with any items changed as per class overrides
sub getEffectiveInterfaceTrafficClass2
{
my $interfaceTrafficClassID = shift;
my $interfaceTrafficClass;
if (!defined($interfaceTrafficClass = getInterfaceTrafficClass2($interfaceTrafficClassID))) {
return;
}
my $realInterfaceTrafficClass = $globals->{'InterfaceTrafficClasses'}->{$interfaceTrafficClassID};
# If we have applied class overrides, check out what changes there may be
if (defined(my $appliedClassOverrides = $realInterfaceTrafficClass->{'.applied_overrides'})) {
my $interfaceTrafficClassOverrideSet;
# Loop with class overrides in ascending fashion, least matches to most
foreach my $interfaceTrafficClassID (
sort { $appliedClassOverrides->{$a} <=> $appliedClassOverrides->{$b} } keys %{$appliedClassOverrides}
) {
my $interfaceTrafficClassOverride = $appliedClassOverrides->{$interfaceTrafficClassID};
# Loop with attributes and create our override set
foreach my $attr (CLASS_OVERRIDE_CHANGESET_ATTRIBUTES) {
# Set class override set attribute if the class override has defined it
if (defined($interfaceTrafficClassOverride->{$attr}) && $interfaceTrafficClassOverride->{$attr} ne "") {
$interfaceTrafficClassOverrideSet->{$attr} = $interfaceTrafficClassOverride->{$attr};
}
}
}
# Set class overrides on pool
if (defined($interfaceTrafficClassOverrideSet)) {
foreach my $attr (keys %{$interfaceTrafficClassOverrideSet}) {
$interfaceTrafficClass->{$attr} = $interfaceTrafficClassOverrideSet->{$attr};
}
}
}
return $interfaceTrafficClass;
}
# Function to set interface traffic class shaper state
sub setInterfaceTrafficClassShaperState
{
my ($interfaceTrafficClassID,$state) = @_;
# Check interface traffic class exists first
if (!isInterfaceTrafficClassIDValid2($interfaceTrafficClassID)) {
return;
}
$globals->{'InterfacesTrafficClasses'}->{$interfaceTrafficClassID}->{'.shaper_state'} |= $state;
return $globals->{'InterfacesTrafficClasses'}->{$interfaceTrafficClassID}->{'.shaper_state'};
}
# Function to unset interface traffic class shaper state
sub unsetInterfaceTrafficClassShaperState
{
my ($interfaceTrafficClassID,$state) = @_;
# Check interface traffic class exists first
if (!isInterfaceTrafficClassIDValid2($interfaceTrafficClassID)) {
return; return;
} }
removePoolMember($pmid); $globals->{'InterfacesTrafficClasses'}->{$interfaceTrafficClassID}->{'.shaper_state'} &= ~$state;
return $globals->{'InterfacesTrafficClasses'}->{$interfaceTrafficClassID}->{'.shaper_state'};
} }
# Event for 'poolmember_change'
sub _session_poolmember_change # Function to get shaper state for a interface traffic class
sub getInterfaceTrafficClassShaperState
{ {
my ($kernel, $poolMemberData) = @_[KERNEL, ARG0]; my $interfaceTrafficClassID = shift;
if (!isPoolMemberIDValid($poolMemberData->{'ID'})) { # Check interface traffic class exists first
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid pool member ID '%s' for 'poolmember_change' event", if (!isInterfaceTrafficClassIDValid2($interfaceTrafficClassID)) {
prettyUndef($poolMemberData->{'ID'})
);
return; return;
} }
changePoolMember($poolMemberData); return $globals->{'InterfacesTrafficClasses'}->{$interfaceTrafficClassID}->{'.shaper_state'};
} }
# Event for 'limit_add'
sub _session_limit_add # Function to get all traffic classes
sub getAllTrafficClasses
{ {
my ($kernel, $limitData) = @_[KERNEL, ARG0]; return ( keys %{$globals->{'TrafficClasses'}} );
}
if (!defined($limitData)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No limit data provided for 'limit_add' event");
return;
}
# Check if we have all the attributes we need # Function to get a traffic class
my $isInvalid; sub getTrafficClass
foreach my $attr (LIMIT_REQUIRED_ATTRIBUTES) { {
if (!defined($limitData->{$attr})) { my $trafficClassID = shift;
$isInvalid = $attr;
last;
} if (!isTrafficClassIDValid($trafficClassID)) {
}
if ($isInvalid) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process limit add as there is an attribute missing: '%s'",$isInvalid);
return; return;
} }
createLimit($limitData); return $globals->{'TrafficClasses'}->{$trafficClassID};
} }
# Event for 'override_add'
sub _session_override_add
{
my ($kernel, $overrideData) = @_[KERNEL, ARG0];
# Function to check if traffic class is valid
sub isTrafficClassIDValid
{
my $trafficClassID = shift;
if (!defined($overrideData)) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] No override data provided for 'override_add' event");
return;
}
# Check that we have at least one match attribute if (!defined($trafficClassID) || !defined($globals->{'TrafficClasses'}->{$trafficClassID})) {
my $isValid = 0;
foreach my $item (OVERRIDE_MATCH_ATTRIBUTES) {
$isValid++;
}
if (!$isValid) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process override as there is no selection attribute");
return; return;
} }
createOverride($overrideData); return $trafficClassID;
} }
# Event for 'override_remove'
sub _session_override_remove # Function to return the traffic priority based on a traffic class
sub getTrafficClassPriority
{ {
my ($kernel, $oid) = @_[KERNEL, ARG0]; my $trafficClassID = shift;
if (!isOverrideIDValid($oid)) { # Check it exists first
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid override ID '%s' for 'override_remove' event",prettyUndef($oid)); if (!isTrafficClassIDValid($trafficClassID)) {
return; return;
} }
removeOverride($oid); # NK: Short circuit, our TrafficClassID = Priority
return $trafficClassID;
} }
# Event for 'override_change'
sub _session_override_change # Function to create an interface
sub createInterface
{ {
my ($kernel, $overrideData) = @_[KERNEL, ARG0]; my $interfaceData = shift;
if (!isOverrideIDValid($overrideData->{'ID'})) { my $interface;
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Invalid override ID '%s' for 'override_change' event",
prettyUndef($overrideData->{'ID'}) # Check if ID is valid
); if (!defined($interface->{'ID'} = $interfaceData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as ID is invalid");
return; return;
} }
changeOverride($overrideData); # Check if Interface is valid
} if (!defined($interface->{'Device'} = $interfaceData->{'Device'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as Device is invalid");
return;
}
# Check if Name is valid
if (!defined($interface->{'Name'} = $interfaceData->{'Name'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as Name is invalid");
return;
}
# Function to check the group ID exists # Check Limit is valid
sub isGroupIDValid if (!defined($interface->{'Limit'} = isNumber($interfaceData->{'Limit'}))) {
{ $logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface as Limit is invalid");
my $gid = shift; return;
}
# Add interface
$globals->{'Interfaces'}->{$interface->{'ID'}} = $interface;
if (defined($config->{'groups'}->{$gid})) { # Create interface main traffic class
return $gid; createInterfaceTrafficClass({
} 'InterfaceID' => $interface->{'ID'},
'TrafficClassID' => 0,
'CIR' => $interfaceData->{'Limit'},
'Limit' => $interfaceData->{'Limit'},
});
return; return $interface->{'ID'};
} }
# Function to return if an interface ID is valid # Function to return if an interface ID is valid
sub isInterfaceIDValid sub isInterfaceIDValid
{ {
my $iid = shift; my $interfaceID = shift;
# Return undef if interface is not valid # Return undef if interface is not valid
if (!defined($config->{'interfaces'}->{$iid})) { if (!defined($globals->{'Interfaces'}->{$interfaceID})) {
return; return;
} }
return $iid; return $interfaceID;
} }
# Function to return the configured Interfaces # Function to return the configured Interfaces
sub getInterfaces sub getInterfaces
{ {
return [ keys %{$config->{'interfaces'}} ]; return ( keys %{$globals->{'Interfaces'}} );
} }
# Return interface classes # Return interface classes
sub getInterface sub getInterface
{ {
my $iid = shift; my $interfaceID = shift;
# If we have this interface return its classes # Check if interface ID is valid
if (!isInterfaceIDValid($iid)) { if (!isInterfaceIDValid($interfaceID)) {
return; return;
} }
my $res = dclone($config->{'interfaces'}->{$iid}); my $res = dclone($globals->{'Interfaces'}->{$interfaceID});
# We don't really want to return classes # We don't want to return TrafficClasses
delete($res->{'classes'}); delete($res->{'TrafficClasses'});
# And return it... # And return it...
return $res; return $res;
} }
# Return interface traffic classes
sub getInterfaceTrafficClasses
{
my $iid = shift;
# If we have this interface return its classes
if (!isInterfaceIDValid($iid)) {
return;
}
return dclone($config->{'interfaces'}->{$iid}->{'classes'});
}
# Function to return our default pool configuration # Function to return our default pool configuration
sub getInterfaceDefaultPool sub getInterfaceDefaultPool
...@@ -1558,60 +2228,88 @@ sub getInterfaceDefaultPool ...@@ -1558,60 +2228,88 @@ sub getInterfaceDefaultPool
# We don't really need the interface to return the default pool # We don't really need the interface to return the default pool
return $config->{'default_pool'}; return $globals->{'DefaultPool'};
} }
# Function to return interface rate
sub getInterfaceRate # Function to create an interface group
sub createInterfaceGroup
{ {
my $iid = shift; my $interfaceGroupData = shift;
my $interfaceGroup;
# If we have this interface return its classes # Check if Name is valid
if (!isInterfaceIDValid($iid)) { if (!defined($interfaceGroup->{'Name'} = $interfaceGroupData->{'Name'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface group as Name is invalid");
return; return;
} }
return $config->{'interfaces'}->{$iid}->{'rate'}; # Check if TxInterface is valid
if (!defined($interfaceGroup->{'TxInterface'} = isInterfaceIDValid($interfaceGroupData->{'TxInterface'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface group as TxInterface is invalid");
return;
}
# Check if RxInterface is valid
if (!defined($interfaceGroup->{'RxInterface'} = isInterfaceIDValid($interfaceGroupData->{'RxInterface'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Failed to add interface group as RxInterface is invalid");
return;
}
$interfaceGroup->{'ID'} = sprintf('%s,%s',$interfaceGroup->{'TxInterface'},$interfaceGroup->{'RxInterface'});
$interfaceGroup->{'IPMap'} = { };
# Add interface group
$globals->{'InterfaceGroups'}->{$interfaceGroup->{'ID'}} = $interfaceGroup;
return $interfaceGroup->{'ID'};
} }
# Function to get interface groups # Function to get interface groups
sub getInterfaceGroups sub getInterfaceGroups
{ {
my $interface_groups = dclone($config->{'interface_groups'}); return ( keys %{$globals->{'InterfaceGroups'}} );
return $interface_groups;
} }
# Function to check if interface group is valid
sub isInterfaceGroupIDValid # Function to get an interface group
sub getInterfaceGroup
{ {
my $igid = shift; my $interfaceGroupID = shift;
if (!defined($igid) || !defined($config->{'interface_groups'}->{$igid})) { if (!isInterfaceGroupIDValid($interfaceGroupID)) {
return; return;
} }
return $igid; my $interfaceGroup = dclone($globals->{'InterfaceGroups'}->{$interfaceGroupID});
delete($interfaceGroup->{'IPMap'});
return $interfaceGroup;
} }
# Function to get an interface group
sub getInterfaceGroup # Function to check if interface group is valid
sub isInterfaceGroupIDValid
{ {
my $igid = shift; my $interfaceGroupID = shift;
if (!isInterfaceGroupIDValid($igid)) { if (!defined($interfaceGroupID) || !defined($globals->{'InterfaceGroups'}->{$interfaceGroupID})) {
return; return;
} }
return dclone($config->{'interface_groups'}->{$igid}); return $interfaceGroupID;
} }
...@@ -1622,6 +2320,7 @@ sub getMatchPriorities ...@@ -1622,6 +2320,7 @@ sub getMatchPriorities
} }
# Function to check if interface group is valid # Function to check if interface group is valid
sub isMatchPriorityIDValid sub isMatchPriorityIDValid
{ {
...@@ -1637,6 +2336,7 @@ sub isMatchPriorityIDValid ...@@ -1637,6 +2336,7 @@ sub isMatchPriorityIDValid
} }
# Function to set a pool attribute # Function to set a pool attribute
sub setPoolAttribute sub setPoolAttribute
{ {
...@@ -1648,12 +2348,13 @@ sub setPoolAttribute ...@@ -1648,12 +2348,13 @@ sub setPoolAttribute
return; return;
} }
$pools->{$pid}->{'.attributes'}->{$attr} = $value; $globals->{'Pools'}->{$pid}->{'.attributes'}->{$attr} = $value;
return $value; return $value;
} }
# Function to get a pool attribute # Function to get a pool attribute
sub getPoolAttribute sub getPoolAttribute
{ {
...@@ -1666,14 +2367,18 @@ sub getPoolAttribute ...@@ -1666,14 +2367,18 @@ sub getPoolAttribute
} }
# Check if attribute exists first # Check if attribute exists first
if (!defined($pools->{$pid}->{'.attributes'}) || !defined($pools->{$pid}->{'.attributes'}->{$attr})) { if (
!defined($globals->{'Pools'}->{$pid}->{'.attributes'}) ||
!defined($globals->{'Pools'}->{$pid}->{'.attributes'}->{$attr}))
{
return; return;
} }
return $pools->{$pid}->{'.attributes'}->{$attr}; return $globals->{'Pools'}->{$pid}->{'.attributes'}->{$attr};
} }
# Function to remove a pool attribute # Function to remove a pool attribute
sub removePoolAttribute sub removePoolAttribute
{ {
...@@ -1686,37 +2391,43 @@ sub removePoolAttribute ...@@ -1686,37 +2391,43 @@ sub removePoolAttribute
} }
# Check if attribute exists first # Check if attribute exists first
if (!defined($pools->{$pid}->{'.attributes'}) || !defined($pools->{$pid}->{'.attributes'}->{$attr})) { if (
!defined($globals->{'Pools'}->{$pid}->{'.attributes'}) ||
!defined($globals->{'Pools'}->{$pid}->{'.attributes'}->{$attr}))
{
return; return;
} }
return delete($pools->{$pid}->{'.attributes'}->{$attr}); return delete($globals->{'Pools'}->{$pid}->{'.attributes'}->{$attr});
} }
# Function to return a override
sub getOverride # Function to return a pool override
sub getPoolOverride
{ {
my $oid = shift; my $poid = shift;
if (!isOverrideIDValid($oid)) { if (!isPoolOverrideIDValid($poid)) {
return; return;
} }
my $override = dclone($overrides->{$oid}); my $poolOverride = dclone($globals->{'PoolOverrides'}->{$poid});
return $override; return $poolOverride;
} }
## Function to return a list of override ID's
sub getOverrides ## Function to return a list of pool override ID's
sub getPoolOverrides
{ {
return (keys %{$overrides}); return (keys %{$globals->{'PoolOverrides'}});
} }
# Function to create a pool # Function to create a pool
sub createPool sub createPool
{ {
...@@ -1745,55 +2456,55 @@ sub createPool ...@@ -1745,55 +2456,55 @@ sub createPool
# Now check if the name is valid # Now check if the name is valid
if (!defined($pool->{'Name'} = $poolData->{'Name'})) { if (!defined($pool->{'Name'} = $poolData->{'Name'})) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool add as Name is invalid"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add as Name is invalid");
return; return;
} }
# Check interface group ID is OK # Check interface group ID is OK
if (!defined($pool->{'InterfaceGroupID'} = isInterfaceGroupIDValid($poolData->{'InterfaceGroupID'}))) { if (!defined($pool->{'InterfaceGroupID'} = isInterfaceGroupIDValid($poolData->{'InterfaceGroupID'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool add for '%s' as the InterfaceGroupID is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the InterfaceGroupID is invalid",
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
# If we already have this name added, return it as the pool # If we already have this name added, return it as the pool
if (defined(my $pool = $poolNameMap->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}})) { if (defined(my $pool = $globals->{'PoolNameMap'}->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}})) {
return $pool->{'ID'}; return $pool->{'ID'};
} }
# Check class is OK # Check class is OK
if (!defined($pool->{'ClassID'} = isTrafficClassIDValid($poolData->{'ClassID'}))) { if (!defined($pool->{'TrafficClassID'} = isTrafficClassIDValid($poolData->{'TrafficClassID'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool add for '%s' as the ClassID is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the TrafficClassID is invalid",
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
# Make sure things are not attached to the default pool # Make sure things are not attached to the default pool
if (defined($config->{'default_pool'}) && $pool->{'ClassID'} eq $config->{'default_pool'}) { if (defined($globals->{'DefaultPool'}) && $pool->{'TrafficClassID'} eq $globals->{'DefaultPool'}) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the ClassID is the 'default_pool' ClassID", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the TrafficClassID is the default pool class",
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
# Check traffic limits # Check traffic limits
if (!isNumber($pool->{'TrafficLimitTx'} = $poolData->{'TrafficLimitTx'})) { if (!isNumber($pool->{'TxCIR'} = $poolData->{'TxCIR'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the TrafficLimitTx is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the TxCIR is invalid",
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
if (!isNumber($pool->{'TrafficLimitRx'} = $poolData->{'TrafficLimitRx'})) { if (!isNumber($pool->{'RxCIR'} = $poolData->{'RxCIR'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the TrafficLimitRx is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool add for '%s' as the RxCIR is invalid",
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
# If we don't have burst limits, improvize # If we don't have burst limits, improvize
if (!defined($pool->{'TrafficLimitTxBurst'} = $poolData->{'TrafficLimitTxBurst'})) { if (!defined($pool->{'TxLimit'} = $poolData->{'TxLimit'})) {
$pool->{'TrafficLimitTxBurst'} = $pool->{'TrafficLimitTx'}; $pool->{'TxLimit'} = $pool->{'TxCIR'};
$pool->{'TrafficLimitTx'} = int($pool->{'TrafficLimitTxBurst'}/4); $pool->{'TxCIR'} = int($pool->{'TxLimit'}/4);
} }
if (!defined($pool->{'TrafficLimitRxBurst'} = $poolData->{'TrafficLimitRxBurst'})) { if (!defined($pool->{'RxLimit'} = $poolData->{'RxLimit'})) {
$pool->{'TrafficLimitRxBurst'} = $pool->{'TrafficLimitRx'}; $pool->{'RxLimit'} = $pool->{'RxCIR'};
$pool->{'TrafficLimitRx'} = int($pool->{'TrafficLimitRxBurst'}/4); $pool->{'RxCIR'} = int($pool->{'RxLimit'}/4);
} }
# Set source # Set source
$pool->{'Source'} = $poolData->{'Source'}; $pool->{'Source'} = $poolData->{'Source'};
...@@ -1810,31 +2521,34 @@ sub createPool ...@@ -1810,31 +2521,34 @@ sub createPool
$pool->{'Notes'} = $poolData->{'Notes'}; $pool->{'Notes'} = $poolData->{'Notes'};
# Assign pool ID # Assign pool ID
$pool->{'ID'} = $poolIDCounter++; $pool->{'ID'} = $globals->{'PoolIDCounter'}++;
# NK: Need better pool ID determination, check what ID is available
# Add pool # Add pool
$pools->{$pool->{'ID'}} = $pool; $globals->{'Pools'}->{$pool->{'ID'}} = $pool;
# Link pool name map # Link pool name map
$poolNameMap->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}} = $pool; $globals->{'PoolNameMap'}->{$pool->{'InterfaceGroupID'}}->{$pool->{'Name'}} = $pool;
# Blank our pool member mapping # Blank our pool member mapping
$poolMemberMap->{$pool->{'ID'}} = { }; $globals->{'PoolMemberMap'}->{$pool->{'ID'}} = { };
setPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE); setPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE);
# Pool needs updating # Pool needs updating
$poolChangeQueue->{$pool->{'ID'}} = $pool; $globals->{'PoolChangeQueue'}->{$pool->{'ID'}} = $pool;
# Resolve overrides # Resolve pool overrides
_override_resolve([$pool->{'ID'}]); _resolve_pool_override([$pool->{'ID'}]);
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return $pool->{'ID'}; return $pool->{'ID'};
} }
# Function to remove a pool # Function to remove a pool
sub removePool sub removePool
{ {
...@@ -1846,7 +2560,7 @@ sub removePool ...@@ -1846,7 +2560,7 @@ sub removePool
return; return;
} }
my $pool = $pools->{$pid}; my $pool = $globals->{'Pools'}->{$pid};
# Check if pool is not already offlining # Check if pool is not already offlining
if ($pool->{'Status'} == CFGM_OFFLINE) { if ($pool->{'Status'} == CFGM_OFFLINE) {
...@@ -1862,15 +2576,16 @@ sub removePool ...@@ -1862,15 +2576,16 @@ sub removePool
$pool->{'LastUpdate'} = $now; $pool->{'LastUpdate'} = $now;
# Pool needs updating # Pool needs updating
$poolChangeQueue->{$pool->{'ID'}} = $pool; $globals->{'PoolChangeQueue'}->{$pool->{'ID'}} = $pool;
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return; return;
} }
# Function to change a pool # Function to change a pool
sub changePool sub changePool
{ {
...@@ -1883,7 +2598,7 @@ sub changePool ...@@ -1883,7 +2598,7 @@ sub changePool
return; return;
} }
my $pool = $pools->{$poolData->{'ID'}}; my $pool = $globals->{'Pools'}->{$poolData->{'ID'}};
my $now = time(); my $now = time();
...@@ -1899,16 +2614,17 @@ sub changePool ...@@ -1899,16 +2614,17 @@ sub changePool
$pool->{'LastUpdate'} = $now; $pool->{'LastUpdate'} = $now;
# Pool needs updating # Pool needs updating
$poolChangeQueue->{$pool->{'ID'}} = $pool; $globals->{'PoolChangeQueue'}->{$pool->{'ID'}} = $pool;
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
# Return what was changed # Return what was changed
return dclone($changes); return dclone($changes);
} }
# Function to return a pool # Function to return a pool
sub getPool sub getPool
{ {
...@@ -1919,16 +2635,17 @@ sub getPool ...@@ -1919,16 +2635,17 @@ sub getPool
return; return;
} }
my $pool = dclone($pools->{$pid}); my $pool = dclone($globals->{'Pools'}->{$pid});
# Remove attributes? # Remove attributes?
delete($pool->{'.attributes'}); delete($pool->{'.attributes'});
delete($pool->{'.applied_attributes'}); delete($pool->{'.applied_overrides'});
return $pool; return $pool;
} }
# Function to get a pool by its name # Function to get a pool by its name
sub getPoolByName sub getPoolByName
{ {
...@@ -1941,21 +2658,26 @@ sub getPoolByName ...@@ -1941,21 +2658,26 @@ sub getPoolByName
} }
# Maybe it doesn't exist? # Maybe it doesn't exist?
if (!defined($poolNameMap->{$interfaceGroupID}) || !defined($poolNameMap->{$interfaceGroupID}->{$name})) { if (
!defined($globals->{'PoolNameMap'}->{$interfaceGroupID}) ||
!defined($globals->{'PoolNameMap'}->{$interfaceGroupID}->{$name}))
{
return; return;
} }
return dclone($poolNameMap->{$interfaceGroupID}->{$name}); return dclone($globals->{'PoolNameMap'}->{$interfaceGroupID}->{$name});
} }
# Function to return a list of pool ID's # Function to return a list of pool ID's
sub getPools sub getPools
{ {
return (keys %{$pools}); return (keys %{$globals->{'Pools'}});
} }
# Function to return a pool TX interface # Function to return a pool TX interface
sub getPoolTxInterface sub getPoolTxInterface
{ {
...@@ -1967,10 +2689,11 @@ sub getPoolTxInterface ...@@ -1967,10 +2689,11 @@ sub getPoolTxInterface
return; return;
} }
return $config->{'interface_groups'}->{$pools->{$pid}->{'InterfaceGroupID'}}->{'txiface'}; return $globals->{'InterfaceGroups'}->{$globals->{'Pools'}->{$pid}->{'InterfaceGroupID'}}->{'TxInterface'};
} }
# Function to return a pool RX interface # Function to return a pool RX interface
sub getPoolRxInterface sub getPoolRxInterface
{ {
...@@ -1982,10 +2705,11 @@ sub getPoolRxInterface ...@@ -1982,10 +2705,11 @@ sub getPoolRxInterface
return; return;
} }
return $config->{'interface_groups'}->{$pools->{$pid}->{'InterfaceGroupID'}}->{'rxiface'}; return $globals->{'InterfaceGroups'}->{$globals->{'Pools'}->{$pid}->{'InterfaceGroupID'}}->{'RxInterface'};
} }
# Function to return a pool traffic class ID # Function to return a pool traffic class ID
sub getPoolTrafficClassID sub getPoolTrafficClassID
{ {
...@@ -1997,10 +2721,11 @@ sub getPoolTrafficClassID ...@@ -1997,10 +2721,11 @@ sub getPoolTrafficClassID
return; return;
} }
return $pools->{$pid}->{'ClassID'}; return $globals->{'Pools'}->{$pid}->{'TrafficClassID'};
} }
# Function to set pools shaper state # Function to set pools shaper state
sub setPoolShaperState sub setPoolShaperState
{ {
...@@ -2012,12 +2737,13 @@ sub setPoolShaperState ...@@ -2012,12 +2737,13 @@ sub setPoolShaperState
return; return;
} }
$pools->{$pid}->{'.shaper_state'} |= $state; $globals->{'Pools'}->{$pid}->{'.shaper_state'} |= $state;
return $pools->{$pid}->{'.shaper_state'}; return $globals->{'Pools'}->{$pid}->{'.shaper_state'};
} }
# Function to unset pools shaper state # Function to unset pools shaper state
sub unsetPoolShaperState sub unsetPoolShaperState
{ {
...@@ -2029,12 +2755,13 @@ sub unsetPoolShaperState ...@@ -2029,12 +2755,13 @@ sub unsetPoolShaperState
return; return;
} }
$pools->{$pid}->{'.shaper_state'} ^= $state; $globals->{'Pools'}->{$pid}->{'.shaper_state'} &= ~$state;
return $pools->{$pid}->{'.shaper_state'}; return $globals->{'Pools'}->{$pid}->{'.shaper_state'};
} }
# Function to get shaper state for a pool # Function to get shaper state for a pool
sub getPoolShaperState sub getPoolShaperState
{ {
...@@ -2046,17 +2773,18 @@ sub getPoolShaperState ...@@ -2046,17 +2773,18 @@ sub getPoolShaperState
return; return;
} }
return $pools->{$pid}->{'.shaper_state'}; return $globals->{'Pools'}->{$pid}->{'.shaper_state'};
} }
# Function to check the pool ID exists # Function to check the pool ID exists
sub isPoolIDValid sub isPoolIDValid
{ {
my $pid = shift; my $pid = shift;
if (!defined($pid) || !defined($pools->{$pid})) { if (!defined($pid) || !defined($globals->{'Pools'}->{$pid})) {
return; return;
} }
...@@ -2064,6 +2792,7 @@ sub isPoolIDValid ...@@ -2064,6 +2792,7 @@ sub isPoolIDValid
} }
# Function to return if a pool is ready for any kind of modification # Function to return if a pool is ready for any kind of modification
sub isPoolReady sub isPoolReady
{ {
...@@ -2076,11 +2805,32 @@ sub isPoolReady ...@@ -2076,11 +2805,32 @@ sub isPoolReady
return; return;
} }
return ($pools->{$pid}->{'Status'} == CFGM_ONLINE && $state & SHAPER_LIVE); return ($globals->{'Pools'}->{$pid}->{'Status'} == CFGM_ONLINE && $state & SHAPER_LIVE);
} }
# Function to return a pool with any items changed as per overrides
# Function to check if pool is being overridden or not
sub isPoolOverridden
{
my $pid = shift;
if (!isPoolIDValid($pid)) {
return;
}
# Set a property based on if this pool is overridden or not
if (defined($globals->{'Pools'}->{$pid}->{'.applied_overrides'}) &&
(keys %{$globals->{'Pools'}->{$pid}->{'.applied_overrides'}}) > 0) {
return 1;
}
return 0;
}
# Function to return a pool with any items changed as per pool overrides
sub getEffectivePool sub getEffectivePool
{ {
my $pid = shift; my $pid = shift;
...@@ -2091,27 +2841,29 @@ sub getEffectivePool ...@@ -2091,27 +2841,29 @@ sub getEffectivePool
return; return;
} }
# If we have applied overrides, check out what changes there may be my $realPool = $globals->{'Pools'}->{$pid};
if (defined(my $appliedOverrides = $pools->{$pid}->{'.applied_overrides'})) {
my $overrideSet;
# Loop with overrides in ascending fashion, least matches to most # If we have applied pool overrides, check out what changes there may be
foreach my $oid ( sort { $appliedOverrides->{$a} <=> $appliedOverrides->{$b} } keys %{$appliedOverrides}) { if (defined(my $appliedPoolOverrides = $realPool->{'.applied_overrides'})) {
my $override = $overrides->{$oid}; my $poolOverrideSet;
# Loop with attributes and create our override set # Loop with pool overrides in ascending fashion, least matches to most
foreach my $attr (OVERRIDE_CHANGESET_ATTRIBUTES) { foreach my $poid ( sort { $appliedPoolOverrides->{$a} <=> $appliedPoolOverrides->{$b} } keys %{$appliedPoolOverrides}) {
# Set override set attribute if the override has defined it my $poolOverride = $globals->{'PoolOverrides'}->{$poid};
if (defined($override->{$attr}) && $override->{$attr} ne "") {
$overrideSet->{$attr} = $override->{$attr}; # Loop with attributes and create our pool override set
foreach my $attr (POOL_OVERRIDE_CHANGESET_ATTRIBUTES) {
# Set pool override set attribute if the pool override has defined it
if (defined($poolOverride->{$attr}) && $poolOverride->{$attr} ne "") {
$poolOverrideSet->{$attr} = $poolOverride->{$attr};
} }
} }
} }
# Set overrides on pool # Set pool overrides on pool
if (defined($overrideSet)) { if (defined($poolOverrideSet)) {
foreach my $attr (keys %{$overrideSet}) { foreach my $attr (keys %{$poolOverrideSet}) {
$pool->{$attr} = $overrideSet->{$attr}; $pool->{$attr} = $poolOverrideSet->{$attr};
} }
} }
} }
...@@ -2120,8 +2872,9 @@ sub getEffectivePool ...@@ -2120,8 +2872,9 @@ sub getEffectivePool
} }
# Function to create a pool member # Function to create a pool member
sub createPoolMember sub createPoolMember
{ {
my $poolMemberData = shift; my $poolMemberData = shift;
...@@ -2144,37 +2897,46 @@ sub createPoolMember ...@@ -2144,37 +2897,46 @@ sub createPoolMember
my $now = time(); my $now = time();
# Check if IP address is defined # Check if IP address is defined
if (!defined(isIP($poolMember->{'IPAddress'} = $poolMemberData->{'IPAddress'}))) { if (!defined(isIPv46CIDR($poolMember->{'IPAddress'} = $poolMemberData->{'IPAddress'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool member add as the IPAddress is invalid"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add as the IPAddress is invalid");
return; return;
} }
if (defined($poolMemberData->{'IPNATAddress'}) && $poolMemberData->{'IPNATAddress'} ne "") {
if (!defined(isIPv46($poolMember->{'IPNATAddress'} = $poolMemberData->{'IPNATAddress'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add as the IPNATAddress is invalid");
return;
} elsif (defined($poolMemberData->{'IPNATInbound'}) && $poolMemberData->{'IPNATInbound'} eq "yes") {
$poolMember->{'IPNATInbound'} = "yes";
}
}
# Now check if Username its valid # Now check if Username its valid
if (!defined(isUsername($poolMember->{'Username'} = $poolMemberData->{'Username'}))) { if (!defined(isUsername($poolMember->{'Username'} = $poolMemberData->{'Username'}, ISUSERNAME_ALLOW_ATSIGN))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool member add as Username is invalid"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add as Username is invalid");
return; return;
} }
# Check pool ID is OK # Check pool ID is OK
if (!defined($poolMember->{'PoolID'} = isPoolIDValid($poolMemberData->{'PoolID'}))) { if (!defined($poolMember->{'PoolID'} = isPoolIDValid($poolMemberData->{'PoolID'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the PoolID is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the PoolID is invalid",
$poolMemberData->{'Username'} $poolMemberData->{'Username'}
); );
return; return;
} }
# Grab pool # Grab pool
my $pool = $pools->{$poolMember->{'PoolID'}}; my $pool = $globals->{'Pools'}->{$poolMember->{'PoolID'}};
# Check match priority ID is OK # Check match priority ID is OK
if (!defined($poolMember->{'MatchPriorityID'} = isMatchPriorityIDValid($poolMemberData->{'MatchPriorityID'}))) { if (!defined($poolMember->{'MatchPriorityID'} = isMatchPriorityIDValid($poolMemberData->{'MatchPriorityID'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the MatchPriorityID is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the MatchPriorityID is invalid",
$poolMemberData->{'Username'} $poolMemberData->{'Username'}
); );
return; return;
} }
# Check group ID is OK # Check group ID is OK
if (!defined($poolMember->{'GroupID'} = isGroupIDValid($poolMemberData->{'GroupID'}))) { if (!defined($poolMember->{'GroupID'} = isGroupIDValid($poolMemberData->{'GroupID'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the GroupID is invalid", $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool member add for '%s' as the GroupID is invalid",
$poolMemberData->{'Username'} $poolMemberData->{'Username'}
); );
return; return;
...@@ -2194,13 +2956,13 @@ sub createPoolMember ...@@ -2194,13 +2956,13 @@ sub createPoolMember
$poolMember->{'Notes'} = $poolMemberData->{'Notes'}; $poolMember->{'Notes'} = $poolMemberData->{'Notes'};
# Create pool member ID # Create pool member ID
$poolMember->{'ID'} = $poolMemberIDCounter++; $poolMember->{'ID'} = $globals->{'PoolMemberIDCounter'}++;
# Add pool member # Add pool member
$poolMembers->{$poolMember->{'ID'}} = $poolMember; $globals->{'PoolMembers'}->{$poolMember->{'ID'}} = $poolMember;
# Link pool map # Link pool map
$poolMemberMap->{$pool->{'ID'}}->{$poolMember->{'ID'}} = $poolMember; $globals->{'PoolMemberMap'}->{$pool->{'ID'}}->{$poolMember->{'ID'}} = $poolMember;
# Updated pool's last updated timestamp # Updated pool's last updated timestamp
$pool->{'LastUpdate'} = $now; $pool->{'LastUpdate'} = $now;
...@@ -2213,14 +2975,15 @@ sub createPoolMember ...@@ -2213,14 +2975,15 @@ sub createPoolMember
# Check for IP conflicts # Check for IP conflicts
if ( if (
defined($interfaceIPMap->{$pool->{'InterfaceGroupID'}}->{$poolMember->{'IPAddress'}}) && defined($globals->{'InterfaceGroups'}->{$pool->{'InterfaceGroupID'}}->{'IPMap'}->{$poolMember->{'IPAddress'}}) &&
(my @conflicts = keys %{$interfaceIPMap->{$pool->{'InterfaceGroupID'}}->{$poolMember->{'IPAddress'}}}) > 0 (my @conflicts = keys %{$globals->{'InterfaceGroups'}->{$pool->{'InterfaceGroupID'}}->{'IPMap'}
->{$poolMember->{'IPAddress'}}}) > 0
) { ) {
# Loop wiht conflicts and build some log items to use # Loop wiht conflicts and build some log items to use
my @logItems; my @logItems;
foreach my $pmid (@conflicts) { foreach my $pmid (@conflicts) {
my $cPoolMember = $poolMembers->{$pmid}; my $cPoolMember = $globals->{'PoolMembers'}->{$pmid};
my $cPool = $pools->{$cPoolMember->{'PoolID'}}; my $cPool = $globals->{'Pools'}->{$cPoolMember->{'PoolID'}};
push(@logItems,sprintf("Pool:%s/Member:%s",$cPool->{'Name'},$cPoolMember->{'Username'})); push(@logItems,sprintf("Pool:%s/Member:%s",$cPool->{'Name'},$cPoolMember->{'Username'}));
} }
...@@ -2231,27 +2994,29 @@ sub createPoolMember ...@@ -2231,27 +2994,29 @@ sub createPoolMember
join(", ",@logItems) join(", ",@logItems)
); );
# We don't have to add it to the queue, as its in a conflicted state
setPoolMemberShaperState($poolMember->{'ID'},SHAPER_CONFLICT); setPoolMemberShaperState($poolMember->{'ID'},SHAPER_CONFLICT);
# We don't have to add it to the queue
} else { } else {
# Pool member needs updating # Pool member needs updating
$poolMemberChangeQueue->{$poolMember->{'ID'}} = $poolMember; $globals->{'PoolMemberChangeQueue'}->{$poolMember->{'ID'}} = $poolMember;
} }
# Link interface IP address map, we must do the check above FIRST, and that neds the pool to be added to the pool map # Link interface IP address map, we must do the check above FIRST, as that needs the pool to be added to the pool map
$interfaceIPMap->{$pool->{'InterfaceGroupID'}}->{$poolMember->{'IPAddress'}}->{$poolMember->{'ID'}} = $poolMember; $globals->{'InterfaceGroups'}->{$pool->{'InterfaceGroupID'}}->{'IPMap'}->{$poolMember->{'IPAddress'}}
->{$poolMember->{'ID'}} = $poolMember;
# Resolve overrides, there may of been no pool members, now there is one and we may be able to apply an override # Resolve pool overrides, there may of been no pool members, now there is one and we may be able to apply a pool override
_override_resolve([$pool->{'ID'}]); _resolve_pool_override([$pool->{'ID'}]);
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return $poolMember->{'ID'}; return $poolMember->{'ID'};
} }
# Function to remove pool member, this function is actually just going to flag it offline # Function to remove pool member, this function is actually just going to flag it offline
# the offline pool members will be caught by cleanup and removed, we do this because we # the offline pool members will be caught by cleanup and removed, we do this because we
# need the pool member setup in the removal functions, we cannot remove it first, and we # need the pool member setup in the removal functions, we cannot remove it first, and we
...@@ -2266,7 +3031,7 @@ sub removePoolMember ...@@ -2266,7 +3031,7 @@ sub removePoolMember
return; return;
} }
my $poolMember = $poolMembers->{$pmid}; my $poolMember = $globals->{'PoolMembers'}->{$pmid};
# Check if pool member is not already offlining # Check if pool member is not already offlining
if ($poolMember->{'Status'} == CFGM_OFFLINE) { if ($poolMember->{'Status'} == CFGM_OFFLINE) {
...@@ -2276,7 +3041,7 @@ sub removePoolMember ...@@ -2276,7 +3041,7 @@ sub removePoolMember
my $now = time(); my $now = time();
# Grab pool # Grab pool
my $pool = $pools->{$poolMember->{'PoolID'}}; my $pool = $globals->{'Pools'}->{$poolMember->{'PoolID'}};
# Updated pool's last updated timestamp # Updated pool's last updated timestamp
$pool->{'LastUpdate'} = $now; $pool->{'LastUpdate'} = $now;
...@@ -2288,15 +3053,16 @@ sub removePoolMember ...@@ -2288,15 +3053,16 @@ sub removePoolMember
$poolMember->{'LastUpdate'} = $now; $poolMember->{'LastUpdate'} = $now;
# Pool member needs updating # Pool member needs updating
$poolMemberChangeQueue->{$poolMember->{'ID'}} = $poolMember; $globals->{'PoolMemberChangeQueue'}->{$poolMember->{'ID'}} = $poolMember;
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return; return;
} }
# Function to change a pool member # Function to change a pool member
sub changePoolMember sub changePoolMember
{ {
...@@ -2309,8 +3075,8 @@ sub changePoolMember ...@@ -2309,8 +3075,8 @@ sub changePoolMember
return; return;
} }
my $poolMember = $poolMembers->{$poolMemberData->{'ID'}}; my $poolMember = $globals->{'PoolMembers'}->{$poolMemberData->{'ID'}};
my $pool = $pools->{$poolMember->{'PoolID'}}; my $pool = $globals->{'Pools'}->{$poolMember->{'PoolID'}};
my $now = time(); my $now = time();
...@@ -2327,13 +3093,14 @@ sub changePoolMember ...@@ -2327,13 +3093,14 @@ sub changePoolMember
$pool->{'LastUpdate'} = $now; $pool->{'LastUpdate'} = $now;
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
# Return what was changed # Return what was changed
return dclone($changes); return dclone($changes);
} }
# Function to return a list of pool ID's # Function to return a list of pool ID's
sub getPoolMembers sub getPoolMembers
{ {
...@@ -2346,14 +3113,15 @@ sub getPoolMembers ...@@ -2346,14 +3113,15 @@ sub getPoolMembers
} }
# Check our member map is not undefined # Check our member map is not undefined
if (!defined($poolMemberMap->{$pid})) { if (!defined($globals->{'PoolMemberMap'}->{$pid})) {
return; return;
} }
return keys %{$poolMemberMap->{$pid}}; return keys %{$globals->{'PoolMemberMap'}->{$pid}};
} }
# Function to return a pool member # Function to return a pool member
sub getPoolMember sub getPoolMember
{ {
...@@ -2365,7 +3133,7 @@ sub getPoolMember ...@@ -2365,7 +3133,7 @@ sub getPoolMember
return; return;
} }
my $poolMember = dclone($poolMembers->{$pmid}); my $poolMember = dclone($globals->{'PoolMembers'}->{$pmid});
# Remove attributes? # Remove attributes?
delete($poolMember->{'.attributes'}); delete($poolMember->{'.attributes'});
...@@ -2374,6 +3142,7 @@ sub getPoolMember ...@@ -2374,6 +3142,7 @@ sub getPoolMember
} }
# Function to return a list of pool ID's # Function to return a list of pool ID's
sub getPoolMemberByUsernameIP sub getPoolMemberByUsernameIP
{ {
...@@ -2386,13 +3155,13 @@ sub getPoolMemberByUsernameIP ...@@ -2386,13 +3155,13 @@ sub getPoolMemberByUsernameIP
} }
# Check our member map is not undefined # Check our member map is not undefined
if (!defined($poolMemberMap->{$pid})) { if (!defined($globals->{'PoolMemberMap'}->{$pid})) {
return; return;
} }
# Loop with pool members and grab the match, there can only be one as we cannot conflict username and IP # Loop with pool members and grab the match, there can only be one as we cannot conflict username and IP
foreach my $pmid (keys %{$poolMemberMap->{$pid}}) { foreach my $pmid (keys %{$globals->{'PoolMemberMap'}->{$pid}}) {
my $poolMember = $poolMemberMap->{$pid}->{$pmid}; my $poolMember = $globals->{'PoolMemberMap'}->{$pid}->{$pmid};
if ($poolMember->{'Username'} eq $username && $poolMember->{'IPAddress'} eq $ipAddress) { if ($poolMember->{'Username'} eq $username && $poolMember->{'IPAddress'} eq $ipAddress) {
return $pmid; return $pmid;
...@@ -2403,6 +3172,7 @@ sub getPoolMemberByUsernameIP ...@@ -2403,6 +3172,7 @@ sub getPoolMemberByUsernameIP
} }
# Function to return pool member ID's with a certain IP address using an interface group # Function to return pool member ID's with a certain IP address using an interface group
sub getAllPoolMembersByInterfaceGroupIP sub getAllPoolMembersByInterfaceGroupIP
{ {
...@@ -2415,21 +3185,22 @@ sub getAllPoolMembersByInterfaceGroupIP ...@@ -2415,21 +3185,22 @@ sub getAllPoolMembersByInterfaceGroupIP
} }
# Maybe it doesn't exist? # Maybe it doesn't exist?
if (!defined($interfaceIPMap->{$interfaceGroupID}) || !defined($interfaceIPMap->{$interfaceGroupID}->{$ipAddress})) { if (!defined($globals->{'InterfaceGroups'}->{$interfaceGroupID}->{'IPMap'}->{$ipAddress})) {
return; return;
} }
return keys %{$interfaceIPMap->{$interfaceGroupID}->{$ipAddress}}; return keys %{$globals->{'InterfaceGroups'}->{$interfaceGroupID}->{'IPMap'}->{$ipAddress}};
} }
# Function to check the pool member ID exists # Function to check the pool member ID exists
sub isPoolMemberIDValid sub isPoolMemberIDValid
{ {
my $pmid = shift; my $pmid = shift;
if (!defined($pmid) || !defined($poolMembers->{$pmid})) { if (!defined($pmid) || !defined($globals->{'PoolMembers'}->{$pmid})) {
return; return;
} }
...@@ -2437,6 +3208,7 @@ sub isPoolMemberIDValid ...@@ -2437,6 +3208,7 @@ sub isPoolMemberIDValid
} }
# Function to return if a pool member is ready for any kind of modification # Function to return if a pool member is ready for any kind of modification
sub isPoolMemberReady sub isPoolMemberReady
{ {
...@@ -2448,10 +3220,11 @@ sub isPoolMemberReady ...@@ -2448,10 +3220,11 @@ sub isPoolMemberReady
return; return;
} }
return ($poolMembers->{$pmid}->{'Status'} == CFGM_ONLINE && getPoolMemberShaperState($pmid) & SHAPER_LIVE); return ($globals->{'PoolMembers'}->{$pmid}->{'Status'} == CFGM_ONLINE && getPoolMemberShaperState($pmid) & SHAPER_LIVE);
} }
# Function to return a pool member match priority # Function to return a pool member match priority
sub getPoolMemberMatchPriority sub getPoolMemberMatchPriority
{ {
...@@ -2464,10 +3237,11 @@ sub getPoolMemberMatchPriority ...@@ -2464,10 +3237,11 @@ sub getPoolMemberMatchPriority
} }
# NK: No actual mappping yet, we just return the ID # NK: No actual mappping yet, we just return the ID
return $poolMembers->{$pmid}->{'MatchPriorityID'}; return $globals->{'PoolMembers'}->{$pmid}->{'MatchPriorityID'};
} }
# Function to set a pool member attribute # Function to set a pool member attribute
sub setPoolMemberAttribute sub setPoolMemberAttribute
{ {
...@@ -2479,12 +3253,13 @@ sub setPoolMemberAttribute ...@@ -2479,12 +3253,13 @@ sub setPoolMemberAttribute
return; return;
} }
$poolMembers->{$pmid}->{'.attributes'}->{$attr} = $value; $globals->{'PoolMembers'}->{$pmid}->{'.attributes'}->{$attr} = $value;
return $value; return $value;
} }
# Function to set pool member shaper state # Function to set pool member shaper state
sub setPoolMemberShaperState sub setPoolMemberShaperState
{ {
...@@ -2496,12 +3271,13 @@ sub setPoolMemberShaperState ...@@ -2496,12 +3271,13 @@ sub setPoolMemberShaperState
return; return;
} }
$poolMembers->{$pmid}->{'.shaper_state'} |= $state; $globals->{'PoolMembers'}->{$pmid}->{'.shaper_state'} |= $state;
return $poolMembers->{$pmid}->{'.shaper_state'}; return $globals->{'PoolMembers'}->{$pmid}->{'.shaper_state'};
} }
# Function to unset pool member shaper state # Function to unset pool member shaper state
sub unsetPoolMemberShaperState sub unsetPoolMemberShaperState
{ {
...@@ -2513,12 +3289,13 @@ sub unsetPoolMemberShaperState ...@@ -2513,12 +3289,13 @@ sub unsetPoolMemberShaperState
return; return;
} }
$poolMembers->{$pmid}->{'.shaper_state'} ^= $state; $globals->{'PoolMembers'}->{$pmid}->{'.shaper_state'} &= ~$state;
return $poolMembers->{$pmid}->{'.shaper_state'}; return $globals->{'PoolMembers'}->{$pmid}->{'.shaper_state'};
} }
# Function to get shaper state for a pool # Function to get shaper state for a pool
sub getPoolMemberShaperState sub getPoolMemberShaperState
{ {
...@@ -2530,10 +3307,11 @@ sub getPoolMemberShaperState ...@@ -2530,10 +3307,11 @@ sub getPoolMemberShaperState
return; return;
} }
return $poolMembers->{$pmid}->{'.shaper_state'}; return $globals->{'PoolMembers'}->{$pmid}->{'.shaper_state'};
} }
# Function to get a pool member attribute # Function to get a pool member attribute
sub getPoolMemberAttribute sub getPoolMemberAttribute
{ {
...@@ -2546,14 +3324,18 @@ sub getPoolMemberAttribute ...@@ -2546,14 +3324,18 @@ sub getPoolMemberAttribute
} }
# Check if attribute exists first # Check if attribute exists first
if (!defined($poolMembers->{$pmid}->{'.attributes'}) || !defined($poolMembers->{$pmid}->{'.attributes'}->{$attr})) { if (
!defined($globals->{'PoolMembers'}->{$pmid}->{'.attributes'}) ||
!defined($globals->{'PoolMembers'}->{$pmid}->{'.attributes'}->{$attr}))
{
return; return;
} }
return $poolMembers->{$pmid}->{'.attributes'}->{$attr}; return $globals->{'PoolMembers'}->{$pmid}->{'.attributes'}->{$attr};
} }
# Function to remove a pool member attribute # Function to remove a pool member attribute
sub removePoolMemberAttribute sub removePoolMemberAttribute
{ {
...@@ -2566,14 +3348,18 @@ sub removePoolMemberAttribute ...@@ -2566,14 +3348,18 @@ sub removePoolMemberAttribute
} }
# Check if attribute exists first # Check if attribute exists first
if (!defined($poolMembers->{$pmid}->{'.attributes'}) || !defined($poolMembers->{$pmid}->{'.attributes'}->{$attr})) { if (
!defined($globals->{'PoolMembers'}->{$pmid}->{'.attributes'}) ||
!defined($globals->{'PoolMembers'}->{$pmid}->{'.attributes'}->{$attr}))
{
return; return;
} }
return delete($poolMembers->{$pmid}->{'.attributes'}->{$attr}); return delete($globals->{'PoolMembers'}->{$pmid}->{'.attributes'}->{$attr});
} }
# Create a limit, which is the combination of a pool and a pool member # Create a limit, which is the combination of a pool and a pool member
sub createLimit sub createLimit
{ {
...@@ -2594,21 +3380,30 @@ sub createLimit ...@@ -2594,21 +3380,30 @@ sub createLimit
} }
# Check if IP address is defined # Check if IP address is defined
if (!defined(isIP($limitData->{'IPAddress'} = $limitData->{'IPAddress'}))) { if (!defined(isIPv46CIDR($limitData->{'IPAddress'} = $limitData->{'IPAddress'}))) {
$logger->log(LOG_NOTICE,"[CONFIGMANAGER] Cannot process limit add as the IP address is invalid"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process limit add as the IPAddress is invalid");
return; return;
} }
if (defined($limitData->{'IPNATAddress'}) && $limitData->{'IPNATAddress'} ne "") {
if (!defined(isIPv46($limitData->{'IPNATAddress'} = $limitData->{'IPNATAddress'}))) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process limit add as the IPNATAddress is invalid");
return;
}
}
if (defined($limitData->{'IPNATInbound'}) && $limitData->{'IPNATInbound'} eq "yes") {
$limitData->{'IPNATInbound'} = "yes";
}
my $poolName = $limitData->{'Username'}; my $poolName = $limitData->{'Username'};
my $poolData = { my $poolData = {
'FriendlyName' => $limitData->{'IPAddress'}, 'FriendlyName' => $limitData->{'FriendlyName'} || $limitData->{'IPAddress'},
'Name' => $poolName, 'Name' => $poolName,
'InterfaceGroupID' => $limitData->{'InterfaceGroupID'}, 'InterfaceGroupID' => $limitData->{'InterfaceGroupID'},
'ClassID' => $limitData->{'ClassID'}, 'TrafficClassID' => $limitData->{'TrafficClassID'},
'TrafficLimitTx' => $limitData->{'TrafficLimitTx'}, 'TxCIR' => $limitData->{'TxCIR'},
'TrafficLimitTxBurst' => $limitData->{'TrafficLimitTxBurst'}, 'TxLimit' => $limitData->{'TxLimit'},
'TrafficLimitRx' => $limitData->{'TrafficLimitRx'}, 'RxCIR' => $limitData->{'RxCIR'},
'TrafficLimitRxBurst' => $limitData->{'TrafficLimitRxBurst'}, 'RxLimit' => $limitData->{'RxLimit'},
'Expires' => $limitData->{'Expires'}, 'Expires' => $limitData->{'Expires'},
'Notes' => $limitData->{'Notes'}, 'Notes' => $limitData->{'Notes'},
'Source' => $limitData->{'Source'} 'Source' => $limitData->{'Source'}
...@@ -2642,273 +3437,213 @@ sub createLimit ...@@ -2642,273 +3437,213 @@ sub createLimit
} }
# Function to create a override
sub createOverride # Function to create a pool override
sub createPoolOverride
{ {
my $overrideData = shift; my $poolOverrideData = shift;
# Check that we have at least one match attribute # Check that we have at least one match attribute
my $isValid = 0; my $isValid = 0;
foreach my $item (OVERRIDE_MATCH_ATTRIBUTES) { foreach my $item (POOL_OVERRIDE_MATCH_ATTRIBUTES) {
$isValid++; # Bump up $isValid if we have a match attribute
if (defined($poolOverrideData->{$item})) {
$isValid++;
}
} }
if (!$isValid) { if (!$isValid) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process override as there is no selection attribute"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool override as there is no selection attribute");
return; return;
} }
my $override; my $poolOverride;
my $now = time(); my $now = time();
# Pull in attributes # Pull in attributes
foreach my $item (OVERRIDE_ATTRIBUTES) { foreach my $item (POOL_OVERRIDE_ATTRIBUTES) {
$override->{$item} = $overrideData->{$item}; $poolOverride->{$item} = $poolOverrideData->{$item};
} }
# Check group is OK # Check group is OK
if (defined($override->{'GroupID'}) && !isGroupIDValid($override->{'GroupID'})) { if (defined($poolOverride->{'GroupID'}) && !isGroupIDValid($poolOverride->{'GroupID'})) {
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Cannot process override for user '%s', IP '%s', GroupID '%s' as the GroupID is ". $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Cannot process pool override for user '%s', IP '%s', GroupID '%s' as the ".
"invalid", "GroupID is invalid",
prettyUndef($override->{'Username'}), prettyUndef($poolOverride->{'Username'}),
prettyUndef($override->{'IPAddress'}), prettyUndef($poolOverride->{'IPAddress'}),
prettyUndef($override->{'GroupID'}) prettyUndef($poolOverride->{'GroupID'})
); );
return; return;
} }
# Check class is OK # Check class is OK
if (defined($override->{'ClassID'}) && !isTrafficClassIDValid($override->{'ClassID'})) { if (defined($poolOverride->{'TrafficClassID'}) && !isTrafficClassIDValid($poolOverride->{'TrafficClassID'})) {
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Cannot process override for user '%s', IP '%s', GroupID '%s' as the ClassID is ". $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Cannot process pool override for user '%s', IP '%s', GroupID '%s' as the ".
"invalid", "TrafficClassID is invalid",
prettyUndef($override->{'Username'}), prettyUndef($poolOverride->{'Username'}),
prettyUndef($override->{'IPAddress'}), prettyUndef($poolOverride->{'IPAddress'}),
prettyUndef($override->{'GroupID'}) prettyUndef($poolOverride->{'GroupID'})
); );
return; return;
} }
# Set source # Set source
$override->{'Source'} = $overrideData->{'Source'}; $poolOverride->{'Source'} = $poolOverrideData->{'Source'};
# Set when this entry was created # Set when this entry was created
$override->{'Created'} = defined($overrideData->{'Created'}) ? $overrideData->{'Created'} : $now; $poolOverride->{'Created'} = defined($poolOverrideData->{'Created'}) ? $poolOverrideData->{'Created'} : $now;
$override->{'LastUpdate'} = $now; $poolOverride->{'LastUpdate'} = $now;
# Set when this entry expires # Set when this entry expires
$override->{'Expires'} = defined($overrideData->{'Expires'}) ? int($overrideData->{'Expires'}) : 0; $poolOverride->{'Expires'} = defined($poolOverrideData->{'Expires'}) ? int($poolOverrideData->{'Expires'}) : 0;
# Check status is OK # Check status is OK
$override->{'Status'} = CFGM_NEW; $poolOverride->{'Status'} = CFGM_NEW;
# Set friendly name and notes # Set friendly name and notes
$override->{'FriendlyName'} = $overrideData->{'FriendlyName'}; $poolOverride->{'FriendlyName'} = $poolOverrideData->{'FriendlyName'};
# Set notes # Set notes
$override->{'Notes'} = $overrideData->{'Notes'}; $poolOverride->{'Notes'} = $poolOverrideData->{'Notes'};
# Create pool member ID # Create pool member ID
$override->{'ID'} = $overrideIDCounter++; $poolOverride->{'ID'} = $globals->{'PoolOverrideIDCounter'}++;
# Add override # Add pool override
$overrides->{$override->{'ID'}} = $override; $globals->{'PoolOverrides'}->{$poolOverride->{'ID'}} = $poolOverride;
# Resolve overrides # Resolve pool overrides
_override_resolve(undef,[$override->{'ID'}]); _resolve_pool_override(undef,[$poolOverride->{'ID'}]);
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return $override->{'ID'}; return $poolOverride->{'ID'};
} }
# Function to remove an override
sub removeOverride # Function to remove a pool override
sub removePoolOverride
{ {
my $oid = shift; my $poid = shift;
# Check override exists first # Check pool override exists first
if (!isOverrideIDValid($oid)) { if (!isPoolOverrideIDValid($poid)) {
return; return;
} }
my $override = $overrides->{$oid}; my $poolOverride = $globals->{'PoolOverrides'}->{$poid};
# Remove override from pools that have it and trigger a change # Remove pool override from pools that have it and trigger a change
if (defined($override->{'.applied_pools'})) { if (defined($poolOverride->{'.applied_pools'})) {
foreach my $pid (keys %{$override->{'.applied_pools'}}) { foreach my $pid (keys %{$poolOverride->{'.applied_pools'}}) {
my $pool = $pools->{$pid}; my $pool = $globals->{'Pools'}->{$pid};
# Remove overrides from the pool # Remove pool overrides from the pool
delete($pool->{'.applied_overrides'}->{$override->{'ID'}}); delete($pool->{'.applied_overrides'}->{$poolOverride->{'ID'}});
# If the pool is online and live, trigger a change # If the pool is online and live, trigger a change
if ($pool->{'Status'} == CFGM_ONLINE && getPoolShaperState($pid) & SHAPER_LIVE) { if ($pool->{'Status'} == CFGM_ONLINE && getPoolShaperState($pid) & SHAPER_LIVE) {
$poolChangeQueue->{$pool->{'ID'}} = $pool; $globals->{'PoolChangeQueue'}->{$pool->{'ID'}} = $pool;
$pool->{'Status'} = CFGM_CHANGED; $pool->{'Status'} = CFGM_CHANGED;
} }
} }
} }
# Remove override # Remove pool override
delete($overrides->{$override->{'ID'}}); delete($globals->{'PoolOverrides'}->{$poolOverride->{'ID'}});
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
return; return;
} }
# Function to change an override
sub changeOverride # Function to change a pool override
sub changePoolOverride
{ {
my $overrideData = shift; my $poolOverrideData = shift;
# Check override exists first # Check pool override exists first
if (!isOverrideIDValid($overrideData->{'ID'})) { if (!isPoolOverrideIDValid($poolOverrideData->{'ID'})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process override change as there is no 'ID' attribute"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Cannot process pool override change as there is no 'ID' attribute");
return; return;
} }
my $override = $overrides->{$overrideData->{'ID'}}; my $poolOverride = $globals->{'PoolOverrides'}->{$poolOverrideData->{'ID'}};
my $now = time(); my $now = time();
my $changes = getHashChanges($override,$overrideData,[OVERRIDE_CHANGE_ATTRIBUTES]); my $changes = getHashChanges($poolOverride,$poolOverrideData,[POOL_OVERRIDE_CHANGE_ATTRIBUTES]);
# Make changes... # Make changes...
foreach my $item (keys %{$changes}) { foreach my $item (keys %{$changes}) {
$override->{$item} = $changes->{$item}; $poolOverride->{$item} = $changes->{$item};
} }
# Set status to updated # Set status to updated
$override->{'Status'} = CFGM_CHANGED; $poolOverride->{'Status'} = CFGM_CHANGED;
# Set timestamp # Set timestamp
$override->{'LastUpdate'} = $now; $poolOverride->{'LastUpdate'} = $now;
# Resolve overrides to see if any attributes changed, we only do this if it already matches # Resolve pool overrides to see if any attributes changed, we only do this if it already matches
# We do NOT support changing match attributes # We do NOT support changing match attributes
if (defined($override->{'.applied_pools'}) && (my @pids = keys %{$override->{'.applied_pools'}}) > 0) { if (defined($poolOverride->{'.applied_pools'}) && (my @pids = keys %{$poolOverride->{'.applied_pools'}}) > 0) {
_override_resolve([@pids],[$override->{'ID'}]); _resolve_pool_override([@pids],[$poolOverride->{'ID'}]);
} }
# Bump up changes # Bump up changes
$stateChanged++; $globals->{'StateChanged'}++;
# Return what was changed # Return what was changed
return dclone($changes); return dclone($changes);
} }
# Function to check the override ID exists
sub isOverrideIDValid
{
my $oid = shift;
if (!defined($oid) || !defined($overrides->{$oid})) {
return;
}
return $oid;
}
# Function to get traffic classes
sub getTrafficClasses
{
my $classes = dclone($config->{'classes'});
# Remove default pool class if we have one
if (defined(my $classID = $config->{'default_pool'})) {
delete($classes->{$classID});
}
return $classes;
}
# Function to get all traffic classes
sub getAllTrafficClasses
{
my $classes = dclone($config->{'classes'});
return $classes;
}
# Function to get class name
sub getTrafficClassName
{
my $classID = shift;
if (!isTrafficClassIDValid($classID)) {
return;
}
return $config->{'classes'}->{$classID};
}
# Function to check if traffic class is valid # Function to check the pool override ID exists
sub isTrafficClassIDValid sub isPoolOverrideIDValid
{ {
my $classID = shift; my $poid = shift;
if (!defined($classID) || !defined($config->{'classes'}->{$classID})) { if (!defined($poid) || !defined($globals->{'PoolOverrides'}->{$poid})) {
return; return;
} }
return $classID; return $poid;
} }
# Function to return the traffic priority based on a traffic class
sub getTrafficClassPriority
{
my $classID = shift;
# Check it exists first
if (!isTrafficClassIDValid($classID)) {
return;
}
# NK: Short circuit, our ClassID = Priority
return $classID;
}
# #
# Internal functions # Internal functions
# #
# Resolve all overrides or those linked to a pid or oid # Resolve all pool overrides or those linked to a pid or oid
# We take 2 optional argument, which is a single override and a single pool to process # We take 2 optional argument, which is a single pool override and a single pool to process
sub _override_resolve sub _resolve_pool_override
{ {
my ($pids,$oids) = @_; my ($pids,$poids) = @_;
# Hack to intercept and create a single element hash if we get ID's above # Hack to intercept and create a single element hash if we get ID's above
my $poolHash; my $poolHash;
if (defined($pids)) { if (defined($pids)) {
foreach my $pid (@{$pids}) { foreach my $pid (@{$pids}) {
$poolHash->{$pid} = $pools->{$pid}; $poolHash->{$pid} = $globals->{'Pools'}->{$pid};
} }
} else { } else {
$poolHash = $pools; $poolHash = $globals->{'Pools'};
} }
my $overrideHash; my $poolOverrideHash;
if (defined($oids)) { if (defined($poids)) {
foreach my $oid (@{$oids}) { foreach my $poid (@{$poids}) {
$overrideHash->{$oid} = $overrides->{$oid}; $poolOverrideHash->{$poid} = $globals->{'PoolOverrides'}->{$poid};
} }
} else { } else {
$overrideHash = $overrides; $poolOverrideHash = $globals->{'PoolOverrides'};
} }
# Loop with all pools, keep a list of pid's updated # Loop with all pools, keep a list of pid's updated
...@@ -2926,69 +3661,71 @@ sub _override_resolve ...@@ -2926,69 +3661,71 @@ sub _override_resolve
$candidate->{'IPAddress'} = $poolMember->{'IPAddress'}; $candidate->{'IPAddress'} = $poolMember->{'IPAddress'};
$candidate->{'GroupID'} = $poolMember->{'GroupID'}; $candidate->{'GroupID'} = $poolMember->{'GroupID'};
} }
# Loop with all overrides and generate a match list # Loop with all pool overrides and generate a match list
while ((my $oid, my $override) = each(%{$overrideHash})) { while ((my $poid, my $poolOverride) = each(%{$poolOverrideHash})) {
my $numMatches = 0; my $numMatches = 0;
my $numMismatches = 0; my $numMismatches = 0;
# Loop with the attributes and check for a full match # Loop with the attributes and check for a full match
foreach my $attr (OVERRIDE_MATCH_ATTRIBUTES) { foreach my $attr (POOL_OVERRIDE_MATCH_ATTRIBUTES) {
# If this attribute in the pool override is set, then lets check it
# If this attribute in the override is set, then lets check it if (defined($poolOverride->{$attr}) && $poolOverride->{$attr} ne "") {
if (defined($override->{$attr}) && $override->{$attr} ne "") { # Check for match or mismatch, only if candidate attribute is defined
# Check for match or mismatch if (defined($candidate->{$attr})) {
if (defined($candidate->{$attr}) && $candidate->{$attr} eq $override->{$attr}) { if ($candidate->{$attr} eq $poolOverride->{$attr}) {
$numMatches++; $numMatches++;
} else { } else {
$numMismatches++; $numMismatches++;
}
} }
} }
} }
# Setup the match list with what was matched # Setup the match list with what was matched
if ($numMatches && !$numMismatches) { if ($numMatches && !$numMismatches) {
$matchList->{$pid}->{$oid} = $numMatches; $matchList->{$pid}->{$poid} = $numMatches;
} else { } else {
$matchList->{$pid}->{$oid} = undef; $matchList->{$pid}->{$poid} = undef;
} }
} }
} }
# Loop with the match list # Loop with the match list
foreach my $pid (keys %{$matchList}) { foreach my $pid (keys %{$matchList}) {
my $pool = $pools->{$pid}; my $pool = $globals->{'Pools'}->{$pid};
# Original Effective pool # Original Effective pool
my $oePool = getEffectivePool($pid); my $oePool = getEffectivePool($pid);
# Loop with overrides for this pool # Loop with pool overrides for this pool
foreach my $oid (keys %{$matchList->{$pid}}) { foreach my $poid (keys %{$matchList->{$pid}}) {
my $override = $overrides->{$oid}; my $poolOverride = $globals->{'PoolOverrides'}->{$poid};
# If we have a match, record it in pools & overrides # If we have a match, record it in pools & pool overrides
if (defined($matchList->{$pid}->{$oid})) { if (defined($matchList->{$pid}->{$poid})) {
# Setup trakcing of what is applied to what # Setup trakcing of what is applied to what
$overrides->{$oid}->{'.applied_pools'}->{$pid} = $matchList->{$pid}->{$oid}; $globals->{'PoolOverrides'}->{$poid}->{'.applied_pools'}->{$pid} = $matchList->{$pid}->{$poid};
$pools->{$pid}->{'.applied_overrides'}->{$oid} = $matchList->{$pid}->{$oid}; $globals->{'Pools'}->{$pid}->{'.applied_overrides'}->{$poid} = $matchList->{$pid}->{$poid};
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Override '%s' [%s] applied to pool '%s' [%s]", $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Pool override '%s' [%s] applied to pool '%s' [%s]",
$override->{'FriendlyName'}, $poolOverride->{'FriendlyName'},
$override->{'ID'}, $poolOverride->{'ID'},
$pool->{'Name'}, $pool->{'Name'},
$pool->{'ID'} $pool->{'ID'}
); );
# We didn't match, but we may of matched before? # We didn't match, but we may of matched before?
} else { } else {
# There was an override before, so something changed now that there is none # There was a pool override before, so something changed now that there is none
if (defined($pools->{$pid}->{'.applied_overrides'}->{$oid})) { if (defined($globals->{'Pools'}->{$pid}->{'.applied_overrides'}->{$poid})) {
# Remove overrides # Remove pool overrides
delete($pools->{$pid}->{'.applied_overrides'}->{$oid}); delete($globals->{'Pools'}->{$pid}->{'.applied_overrides'}->{$poid});
delete($overrides->{$oid}->{'.applied_pools'}->{$pid}); delete($globals->{'PoolOverrides'}->{$poid}->{'.applied_pools'}->{$pid});
$logger->log(LOG_DEBUG,"[CONFIGMANAGER] Override '%s' no longer applies to pool '%s' [%s]", $logger->log(LOG_DEBUG,"[CONFIGMANAGER] Pool override '%s' no longer applies to pool '%s' [%s]",
$override->{'ID'}, $poolOverride->{'ID'},
$pool->{'Name'}, $pool->{'Name'},
$pool->{'ID'} $pool->{'ID'}
); );
...@@ -2999,22 +3736,23 @@ sub _override_resolve ...@@ -2999,22 +3736,23 @@ sub _override_resolve
my $nePool = getEffectivePool($pid); my $nePool = getEffectivePool($pid);
# Get changes between effective pool states # Get changes between effective pool states
my $poolChanges = getHashChanges($oePool,$nePool,[OVERRIDE_CHANGESET_ATTRIBUTES]); my $poolChanges = getHashChanges($oePool,$nePool,[POOL_OVERRIDE_CHANGESET_ATTRIBUTES]);
# If there were pool changes, trigger a pool update # If there were pool changes, trigger a pool update
if (keys %{$poolChanges} > 0) { if (keys %{$poolChanges} > 0) {
# If the pool is currently online and live, trigger a change # If the pool is currently online and live, trigger a change
if ($pool->{'Status'} == CFGM_ONLINE && getPoolShaperState($pid) & SHAPER_LIVE) { if ($pool->{'Status'} == CFGM_ONLINE && getPoolShaperState($pid) & SHAPER_LIVE) {
$pool->{'Status'} = CFGM_CHANGED; $pool->{'Status'} = CFGM_CHANGED;
$poolChangeQueue->{$pool->{'ID'}} = $pool; $globals->{'PoolChangeQueue'}->{$pool->{'ID'}} = $pool;
} }
} }
} }
} }
# Remove pool override information # Remove pool override information
sub _override_remove_pool sub _remove_pool_override
{ {
my $pid = shift; my $pid = shift;
...@@ -3023,17 +3761,18 @@ sub _override_remove_pool ...@@ -3023,17 +3761,18 @@ sub _override_remove_pool
return; return;
} }
my $pool = $pools->{$pid}; my $pool = $globals->{'Pools'}->{$pid};
# Remove pool from overrides if there are any # Remove pool from pool overrides if there are any
if (defined($pool->{'.applied_overrides'})) { if (defined($pool->{'.applied_overrides'})) {
foreach my $oid (keys %{$pool->{'.applied_overrides'}}) { foreach my $poid (keys %{$pool->{'.applied_overrides'}}) {
delete($overrides->{$oid}->{'.applied_pools'}->{$pool->{'ID'}}); delete($globals->{'PoolOverrides'}->{$poid}->{'.applied_pools'}->{$pool->{'ID'}});
} }
} }
} }
# Load our statefile # Load our statefile
sub _load_statefile sub _load_statefile
{ {
...@@ -3069,24 +3808,44 @@ sub _load_statefile ...@@ -3069,24 +3808,44 @@ sub _load_statefile
# Grab the object handle # Grab the object handle
my $state = tied( %stateHash ); my $state = tied( %stateHash );
# Loop with user overrides # Loop with interface traffic class overrides
foreach my $section ($state->GroupMembers('override')) { foreach my $section ($state->GroupMembers('interface_traffic_class.override')) {
my $override = $stateHash{$section}; my $classOverride = $stateHash{$section};
# Loop with the persistent attributes and create our hash
my $cClassOverride;
foreach my $attr (CLASS_OVERRIDE_PERSISTENT_ATTRIBUTES) {
if (defined($classOverride->{$attr})) {
# If its an array, join all the items
if (ref($classOverride->{$attr}) eq "ARRAY") {
$classOverride->{$attr} = join("\n",@{$classOverride->{$attr}});
}
$cClassOverride->{$attr} = $classOverride->{$attr};
}
}
# XXX - Hack, Proces this class override
changeInterfaceTrafficClass($cClassOverride);
}
# Loop with user pool overrides
foreach my $section ($state->GroupMembers('pool.override')) {
my $poolOverride = $stateHash{$section};
# Loop with the persistent attributes and create our hash # Loop with the persistent attributes and create our hash
my $coverride; my $cPoolOverride;
foreach my $attr (OVERRIDE_PERSISTENT_ATTRIBUTES) { foreach my $attr (POOL_OVERRIDE_PERSISTENT_ATTRIBUTES) {
if (defined($override->{$attr})) { if (defined($poolOverride->{$attr})) {
# If its an array, join all the items # If its an array, join all the items
if (ref($override->{$attr}) eq "ARRAY") { if (ref($poolOverride->{$attr}) eq "ARRAY") {
$override->{$attr} = join("\n",@{$override->{$attr}}); $poolOverride->{$attr} = join("\n",@{$poolOverride->{$attr}});
} }
$coverride->{$attr} = $override->{$attr}; $cPoolOverride->{$attr} = $poolOverride->{$attr};
} }
} }
# Proces this override # Proces this pool override
createOverride($coverride); createPoolOverride($cPoolOverride);
} }
# We need a pool ID translation, when we recreate pools we get different ID's, we cannot restore members with orignal ID's # We need a pool ID translation, when we recreate pools we get different ID's, we cannot restore members with orignal ID's
...@@ -3153,6 +3912,7 @@ sub _load_statefile ...@@ -3153,6 +3912,7 @@ sub _load_statefile
} }
# Write out statefile # Write out statefile
sub _write_statefile sub _write_statefile
{ {
...@@ -3160,8 +3920,8 @@ sub _write_statefile ...@@ -3160,8 +3920,8 @@ sub _write_statefile
# We reset this early so we don't get triggred continuously if we encounter errors # We reset this early so we don't get triggred continuously if we encounter errors
$stateChanged = 0; $globals->{'StateChanged'} = 0;
$lastStateSync = time(); $globals->{'LastStateSync'} = time();
# Check if the state file exists first of all # Check if the state file exists first of all
if (!defined($config->{'statefile'})) { if (!defined($config->{'statefile'})) {
...@@ -3169,9 +3929,9 @@ sub _write_statefile ...@@ -3169,9 +3929,9 @@ sub _write_statefile
return; return;
} }
# Only write out if we actually have limits & overrides, else we may of crashed? # Only write out if we actually have limits & pool overrides, else we may of crashed?
if (!(keys %{$pools}) && !(keys %{$overrides})) { if (!(keys %{$globals->{'Pools'}}) && !(keys %{$globals->{'PoolOverrides'}})) {
$logger->log(LOG_WARN,"[CONFIGMANAGER] Not writing state file as there are no active pools or overrides"); $logger->log(LOG_WARN,"[CONFIGMANAGER] Not writing state file as there are no active pools or pool overrides");
return; return;
} }
...@@ -3182,25 +3942,52 @@ sub _write_statefile ...@@ -3182,25 +3942,52 @@ sub _write_statefile
# Create new state file object # Create new state file object
my $state = new Config::IniFiles(); my $state = new Config::IniFiles();
# Loop with overrides # XXX - Hack, loop with class overrides
while ((my $oid, my $override) = each(%{$overrides})) { while ((my $itcid, my $interfaceTrafficClass) = each(%{$globals->{'InterfaceTrafficClasses'}})) {
# Skip over non-overridden classes
if (!defined($interfaceTrafficClass->{'.applied_overrides'})) {
next;
}
# Create a section name # Create a section name
my $section = "override " . $oid; my $section = "interface_traffic_class.override " . $itcid;
# Add a section for this override # Add a section for this class override
$state->AddSection($section); $state->AddSection($section);
# Attributes we want to save for this override # XXX - Hack, Attributes we want to save for this traffic class override
foreach my $attr (OVERRIDE_PERSISTENT_ATTRIBUTES) { foreach my $attr (CLASS_OVERRIDE_PERSISTENT_ATTRIBUTES) {
# Set items up
if (defined(my $value = $interfaceTrafficClass->{$attr})) {
$state->newval($section,$attr,$value);
}
}
# XXX - Hack, loop with the override
foreach my $attr (CLASS_OVERRIDE_PERSISTENT_ATTRIBUTES) {
# Set items up # Set items up
if (defined(my $value = $overrides->{$oid}->{$attr})) { if (defined(my $value = $interfaceTrafficClass->{'.applied_overrides'}->{'change'}->{$attr})) {
$state->newval($section,$attr,$value); $state->newval($section,$attr,$value);
} }
} }
}
# Loop with pool overrides
while ((my $poid, my $poolOverride) = each(%{$globals->{'PoolOverrides'}})) {
# Create a section name
my $section = "pool.override " . $poid;
# Add a section for this pool override
$state->AddSection($section);
# Attributes we want to save for this pool override
foreach my $attr (POOL_OVERRIDE_PERSISTENT_ATTRIBUTES) {
# Set items up
if (defined(my $value = $globals->{'PoolOverrides'}->{$poid}->{$attr})) {
$state->newval($section,$attr,$value);
}
}
} }
# Loop with pools # Loop with pools
while ((my $pid, my $pool) = each(%{$pools})) { while ((my $pid, my $pool) = each(%{$globals->{'Pools'}})) {
# Skip over dynamic entries, we only want persistent ones unless we doing a full write # Skip over dynamic entries, we only want persistent ones unless we doing a full write
next if (!$fullWrite && $pool->{'Source'} eq "plugin.radius"); next if (!$fullWrite && $pool->{'Source'} eq "plugin.radius");
...@@ -3218,14 +4005,14 @@ sub _write_statefile ...@@ -3218,14 +4005,14 @@ sub _write_statefile
} }
# Save pool members too # Save pool members too
foreach my $pmid (keys %{$poolMemberMap->{$pid}}) { foreach my $pmid (keys %{$globals->{'PoolMemberMap'}->{$pid}}) {
# Create a section name for the pool member # Create a section name for the pool member
$section = "pool_member " . $pmid; $section = "pool_member " . $pmid;
# Add a new section for this pool member # Add a new section for this pool member
$state->AddSection($section); $state->AddSection($section);
my $poolMember = $poolMembers->{$pmid}; my $poolMember = $globals->{'PoolMembers'}->{$pmid};
# Items we want for persistent entries # Items we want for persistent entries
foreach my $attr (POOLMEMBER_PERSISTENT_ATTRIBUTES) { foreach my $attr (POOLMEMBER_PERSISTENT_ATTRIBUTES) {
...@@ -3266,5 +4053,6 @@ sub _write_statefile ...@@ -3266,5 +4053,6 @@ sub _write_statefile
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper radius module # OpenTrafficShaper radius module
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -29,7 +29,7 @@ use POE; ...@@ -29,7 +29,7 @@ use POE;
use IO::Socket::INET; use IO::Socket::INET;
use opentrafficshaper::logger; use opentrafficshaper::logger;
use opentrafficshaper::utils; use awitpt::util;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
createPool createPool
changePool changePool
...@@ -61,7 +61,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -61,7 +61,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
); );
use constant { use constant {
VERSION => '0.2.1', VERSION => '1.0.0',
DATAGRAM_MAXLEN => 8192, DATAGRAM_MAXLEN => 8192,
...@@ -86,9 +86,12 @@ our $pluginInfo = { ...@@ -86,9 +86,12 @@ our $pluginInfo = {
}; };
# Copy of system globals # Our globals
my $globals; my $globals;
# Copy of system logger
my $logger; my $logger;
# Our own data storage # Our own data storage
my $config = { my $config = {
'expiry_period' => DEFAULT_EXPIRY_PERIOD, 'expiry_period' => DEFAULT_EXPIRY_PERIOD,
...@@ -99,32 +102,34 @@ my $config = { ...@@ -99,32 +102,34 @@ my $config = {
'group' => 1, 'group' => 1,
}; };
my $dictionary;
# Initialize plugin # Initialize plugin
sub plugin_init sub plugin_init
{ {
$globals = shift; my $system = shift;
# Setup our environment # Setup our environment
$logger = $globals->{'logger'}; $logger = $system->{'logger'};
$logger->log(LOG_NOTICE,"[RADIUS] OpenTrafficShaper Radius Module v%s - Copyright (c) 2013-2014, AllWorldIT",VERSION); $logger->log(LOG_NOTICE,"[RADIUS] OpenTrafficShaper Radius Module v%s - Copyright (c) 2013-2014, AllWorldIT",VERSION);
# Inititalize
$globals->{'Dictionary'} = undef;
# Split off dictionaries to load # Split off dictionaries to load
my @dicts = ref($globals->{'file.config'}->{'plugin.radius'}->{'dictionary'}) eq "ARRAY" ? my @dicts = ref($system->{'file.config'}->{'plugin.radius'}->{'dictionary'}) eq "ARRAY" ?
@{$globals->{'file.config'}->{'plugin.radius'}->{'dictionary'}} : @{$system->{'file.config'}->{'plugin.radius'}->{'dictionary'}} :
( $globals->{'file.config'}->{'plugin.radius'}->{'dictionary'} ); ( $system->{'file.config'}->{'plugin.radius'}->{'dictionary'} );
foreach my $dict (@dicts) { foreach my $dict (@dicts) {
$dict =~ s/\s+//g; $dict =~ s/\s+//g;
# Skip comments # Skip comments
next if ($dict =~ /^#/); next if ($dict =~ /^#/);
# Check if we have a path, if we do use it # Check if we have a path, if we do use it
if (defined($globals->{'file.config'}->{'plugin.radius'}->{'dictionary_path'})) { if (defined($system->{'file.config'}->{'plugin.radius'}->{'dictionary_path'})) {
$dict = $globals->{'file.config'}->{'plugin.radius'}->{'dictionary_path'} . "/$dict"; $dict = $system->{'file.config'}->{'plugin.radius'}->{'dictionary_path'} . "/$dict";
} }
push(@{$config->{'config.dictionaries'}},$dict); push(@{$config->{'config.dictionaries'}},$dict);
} }
...@@ -142,22 +147,22 @@ sub plugin_init ...@@ -142,22 +147,22 @@ sub plugin_init
} }
$logger->log(LOG_DEBUG,"[RADIUS] Loading dictionaries completed."); $logger->log(LOG_DEBUG,"[RADIUS] Loading dictionaries completed.");
# Store the dictionary # Store the dictionary
$dictionary = $dict; $globals->{'Dictionary'} = $dict;
# Check if we must override the expiry time # Check if we must override the expiry time
if (defined(my $expiry = $globals->{'file.config'}->{'plugin.radius'}->{'expiry_period'})) { if (defined(my $expiry = $system->{'file.config'}->{'plugin.radius'}->{'expiry_period'})) {
$logger->log(LOG_INFO,"[RADIUS] Set expiry_period to '%s'",$expiry); $logger->log(LOG_INFO,"[RADIUS] Set expiry_period to '%s'",$expiry);
$config->{'expiry_period'} = $expiry; $config->{'expiry_period'} = $expiry;
} }
# Check if we got a username to pool transform # Check if we got a username to pool transform
if (defined(my $userPoolTransform = $globals->{'file.config'}->{'plugin.radius'}->{'username_to_pool_transform'})) { if (defined(my $userPoolTransform = $system->{'file.config'}->{'plugin.radius'}->{'username_to_pool_transform'})) {
$logger->log(LOG_INFO,"[RADIUS] Set username_to_pool_transform to '%s'",$userPoolTransform); $logger->log(LOG_INFO,"[RADIUS] Set username_to_pool_transform to '%s'",$userPoolTransform);
$config->{'username_to_pool_transform'} = $userPoolTransform; $config->{'username_to_pool_transform'} = $userPoolTransform;
} }
# Default interface group to use # Default interface group to use
if (defined(my $interfaceGroup = $globals->{'file.config'}->{'plugin.radius'}->{'default_interface_group'})) { if (defined(my $interfaceGroup = $system->{'file.config'}->{'plugin.radius'}->{'default_interface_group'})) {
if (isInterfaceGroupIDValid($interfaceGroup)) { if (isInterfaceGroupIDValid($interfaceGroup)) {
$logger->log(LOG_INFO,"[RADIUS] Set interface_group to '%s'",$interfaceGroup); $logger->log(LOG_INFO,"[RADIUS] Set interface_group to '%s'",$interfaceGroup);
$config->{'interface_group'} = $interfaceGroup; $config->{'interface_group'} = $interfaceGroup;
...@@ -169,7 +174,7 @@ sub plugin_init ...@@ -169,7 +174,7 @@ sub plugin_init
} }
# Default match priority to use # Default match priority to use
if (defined(my $matchPriority = $globals->{'file.config'}->{'plugin.radius'}->{'default_match_priority'})) { if (defined(my $matchPriority = $system->{'file.config'}->{'plugin.radius'}->{'default_match_priority'})) {
if (isMatchPriorityIDValid($matchPriority)) { if (isMatchPriorityIDValid($matchPriority)) {
$logger->log(LOG_INFO,"[RADIUS] Set match_priority to '%s'",$matchPriority); $logger->log(LOG_INFO,"[RADIUS] Set match_priority to '%s'",$matchPriority);
$config->{'match_priority'} = $matchPriority; $config->{'match_priority'} = $matchPriority;
...@@ -179,7 +184,7 @@ sub plugin_init ...@@ -179,7 +184,7 @@ sub plugin_init
} }
# Default traffic class to use # Default traffic class to use
if (defined(my $trafficClassID = $globals->{'file.config'}->{'plugin.radius'}->{'default_traffic_class'})) { if (defined(my $trafficClassID = $system->{'file.config'}->{'plugin.radius'}->{'default_traffic_class'})) {
if (isTrafficClassIDValid($trafficClassID)) { if (isTrafficClassIDValid($trafficClassID)) {
$logger->log(LOG_INFO,"[RADIUS] Set traffic_class to '%s'",$trafficClassID); $logger->log(LOG_INFO,"[RADIUS] Set traffic_class to '%s'",$trafficClassID);
$config->{'traffic_class'} = $trafficClassID; $config->{'traffic_class'} = $trafficClassID;
...@@ -189,7 +194,7 @@ sub plugin_init ...@@ -189,7 +194,7 @@ sub plugin_init
} }
# Default group to use # Default group to use
if (defined(my $group = $globals->{'file.config'}->{'plugin.radius'}->{'default_group'})) { if (defined(my $group = $system->{'file.config'}->{'plugin.radius'}->{'default_group'})) {
if (isGroupIDValid($group)) { if (isGroupIDValid($group)) {
$logger->log(LOG_INFO,"[RADIUS] Set group to '%s'",$group); $logger->log(LOG_INFO,"[RADIUS] Set group to '%s'",$group);
$config->{'group'} = $group; $config->{'group'} = $group;
...@@ -211,6 +216,7 @@ sub plugin_init ...@@ -211,6 +216,7 @@ sub plugin_init
} }
# Start the plugin # Start the plugin
sub plugin_start sub plugin_start
{ {
...@@ -218,6 +224,7 @@ sub plugin_start ...@@ -218,6 +224,7 @@ sub plugin_start
} }
# Initialize server # Initialize server
sub _session_start sub _session_start
{ {
...@@ -245,6 +252,7 @@ sub _session_start ...@@ -245,6 +252,7 @@ sub _session_start
} }
# Shut down server # Shut down server
sub _session_stop sub _session_stop
{ {
...@@ -258,7 +266,6 @@ sub _session_stop ...@@ -258,7 +266,6 @@ sub _session_stop
# Blow everything away # Blow everything away
$globals = undef; $globals = undef;
$dictionary = undef;
$logger->log(LOG_DEBUG,"[RADIUS] Shutdown"); $logger->log(LOG_DEBUG,"[RADIUS] Shutdown");
...@@ -266,6 +273,7 @@ sub _session_stop ...@@ -266,6 +273,7 @@ sub _session_stop
} }
# Read event for server # Read event for server
sub _session_socket_read sub _session_socket_read
{ {
...@@ -285,7 +293,7 @@ sub _session_socket_read ...@@ -285,7 +293,7 @@ sub _session_socket_read
my $peer_addr_h = inet_ntoa($peer_addr); my $peer_addr_h = inet_ntoa($peer_addr);
# Parse packet # Parse packet
my $pkt = opentrafficshaper::plugins::radius::Radius::Packet->new($dictionary,$udp_packet); my $pkt = opentrafficshaper::plugins::radius::Radius::Packet->new($globals->{'Dictionary'},$udp_packet);
# Build log line # Build log line
my $logLine = sprintf("Remote: %s:%s, Code: %s, Identifier: %s => ",$peer_addr_h,$peer_port,$pkt->code,$pkt->identifier); my $logLine = sprintf("Remote: %s:%s, Code: %s, Identifier: %s => ",$peer_addr_h,$peer_port,$pkt->code,$pkt->identifier);
...@@ -354,39 +362,70 @@ sub _session_socket_read ...@@ -354,39 +362,70 @@ sub _session_socket_read
if (my $attrRawVal = $pkt->vsattr(IANA_PEN,'OpenTrafficShaper-Traffic-Limit')) { if (my $attrRawVal = $pkt->vsattr(IANA_PEN,'OpenTrafficShaper-Traffic-Limit')) {
$trafficLimit = @{ $attrRawVal }[0]; $trafficLimit = @{ $attrRawVal }[0];
} }
# Grab rate limits from the string we got # We assume below that we will have limits
my $trafficLimitRx; my $trafficLimitTx; if (!defined($trafficLimit)) {
my $trafficLimitRxBurst; my $trafficLimitTxBurst; $logger->log(LOG_NOTICE,"[RADIUS] No traffic limit set for user '%s', ignoring",$username);
if (defined($trafficLimit)) { return;
# Match rx-rate[/tx-rate] rx-burst-rate[/tx-burst-rate] }
if ($trafficLimit =~ /^(\d+)([km])(?:\/(\d+)([km]))?(?: (\d+)([km])(?:\/(\d+)([km]))?)?/) { # Grab rate limits below from the string we got
$trafficLimitRx = getKbit($1,$2); my $rxCIR; my $txCIR;
$trafficLimitTx = getKbit($3,$4); my $rxLimit; my $txLimit;
$trafficLimitRxBurst = getKbit($5,$6); # Match rx-rate[/tx-rate] rx-burst-rate[/tx-burst-rate]
$trafficLimitTxBurst = getKbit($7,$8); if ($trafficLimit =~ /^(\d+)([km])(?:\/(\d+)([km]))?(?: (\d+)([km])(?:\/(\d+)([km]))?)?/) {
} else { $rxCIR = getKbit($1,$2);
$logger->log(LOG_DEBUG,"[RADIUS] The 'OpenTrafficShaper-Traffic-Limit' attribute appears to be invalid for user '%s'". $txCIR = getKbit($3,$4);
": '%s'", $rxLimit = getKbit($5,$6);
$username, $txLimit = getKbit($7,$8);
$trafficLimit
); # Set our limits if they not defined
return; if (!defined($rxLimit)) {
$rxLimit = $rxCIR;
$rxCIR = $rxCIR / 4;
} }
if (!defined($txLimit)) {
$txLimit = $txCIR;
$txCIR = $txCIR / 4;
}
} else {
$logger->log(LOG_WARN,"[RADIUS] The 'OpenTrafficShaper-Traffic-Limit' attribute appears to be invalid for user '%s'".
": '%s'",
$username,
$trafficLimit
);
return;
} }
# Check if we have a pool transform # Check if we have a pool transform
my $tPoolName; my $poolName;
if (defined($config->{'username_to_pool_transform'})) { if (defined($config->{'username_to_pool_transform'})) {
# Check if transform matches, if it does set pool name # Check if transform matches, if it does set pool name
if ($username =~ $config->{'username_to_pool_transform'}) { if ($username =~ $config->{'username_to_pool_transform'}) {
my $tPoolName = $1; $poolName = $1;
} }
} }
# Check what to use for the pool name, by default its the username # Check if the pool name is being overridden
my $poolName = $tPoolName || $username; if (my $attrRawVal = $pkt->vsattr(IANA_PEN,'OpenTrafficShaper-Traffic-Pool')) {
$poolName = @{ $attrRawVal }[0];
}
# If we got a pool name, check if it exists
if (defined($poolName)) {
if (!defined(getPoolByName($config->{'interface_group'},$poolName))) {
$logger->log(LOG_NOTICE,"[RADIUS] Pool '%s' not found, using username '%s' instead",
$poolName,
$username
);
$poolName = $username;
}
# If we didn't get the pool name, just use the username
} else {
$poolName = $username;
}
# Try grab the pool # Try grab the pool
my $pool = getPoolByName($poolName); my $pool = getPoolByName($config->{'interface_group'},$poolName);
my $pid = defined($pool) ? $pool->{'ID'} : undef; my $pid = defined($pool) ? $pool->{'ID'} : undef;
my $ipAddress = $pkt->attr('Framed-IP-Address'); my $ipAddress = $pkt->attr('Framed-IP-Address');
...@@ -401,10 +440,10 @@ sub _session_socket_read ...@@ -401,10 +440,10 @@ sub _session_socket_read
$config->{'match_priority'}, $config->{'match_priority'},
$group, $group,
$trafficClassID, $trafficClassID,
prettyUndef($trafficLimitTx), prettyUndef($txCIR),
prettyUndef($trafficLimitRx), prettyUndef($rxCIR),
prettyUndef($trafficLimitTxBurst), prettyUndef($txLimit),
prettyUndef($trafficLimitRxBurst) prettyUndef($rxLimit)
); );
# Check if user is new or online # Check if user is new or online
...@@ -420,17 +459,25 @@ sub _session_socket_read ...@@ -420,17 +459,25 @@ sub _session_socket_read
# Change the details # Change the details
my $changes = changePool({ my $changes = changePool({
'ID' => $pid, 'ID' => $pid,
'ClassID' => $trafficClassID, 'FriendlyName' => $ipAddress,
'TrafficLimitTx' => $trafficLimitTx, 'TrafficClassID' => $trafficClassID,
'TrafficLimitRx' => $trafficLimitRx, 'TxCIR' => $txCIR,
'TrafficLimitTxBurst' => $trafficLimitTxBurst, 'RxCIR' => $rxCIR,
'TrafficLimitRxBurst' => $trafficLimitRxBurst, # These MUST be defined
'TxLimit' => $txLimit,
'RxLimit' => $rxLimit,
'Expires' => $now + DEFAULT_EXPIRY_PERIOD 'Expires' => $now + DEFAULT_EXPIRY_PERIOD
}); });
my @txtChanges; my @txtChanges;
foreach my $item (keys %{$changes}) { foreach my $item (keys %{$changes}) {
push(@txtChanges,sprintf("%s = %s",$item,$changes->{$item})); # Make expires look nice
my $value = $changes->{$item};
if ($item eq "Expires") {
$value = sprintf("%s [%s]",$value,scalar(localtime($value)));
}
push(@txtChanges,sprintf("%s = %s",$item,$value));
} }
if (@txtChanges) { if (@txtChanges) {
$logger->log(LOG_INFO,"[RADIUS] Pool '%s' updated: %s",$poolName,join(", ",@txtChanges)); $logger->log(LOG_INFO,"[RADIUS] Pool '%s' updated: %s",$poolName,join(", ",@txtChanges));
...@@ -445,12 +492,12 @@ sub _session_socket_read ...@@ -445,12 +492,12 @@ sub _session_socket_read
# No pool, time to create one # No pool, time to create one
} else { } else {
# If we don't have rate limits, short circuit # If we don't have rate limits, short circuit
if (!defined($trafficLimitTx)) { if (!defined($txCIR)) {
$logger->log(LOG_NOTICE,"[RADIUS] Pool '%s' has no 'TrafficLimitTx', aborting",$poolName); $logger->log(LOG_NOTICE,"[RADIUS] Pool '%s' has no 'TxCIR', aborting",$poolName);
return; return;
} }
if (!defined($trafficLimitRx)) { if (!defined($rxCIR)) {
$logger->log(LOG_NOTICE,"[RADIUS] Pool '%s' has no 'TrafficLimitRx', aborting",$poolName); $logger->log(LOG_NOTICE,"[RADIUS] Pool '%s' has no 'RxCIR', aborting",$poolName);
return; return;
} }
...@@ -459,11 +506,11 @@ sub _session_socket_read ...@@ -459,11 +506,11 @@ sub _session_socket_read
'FriendlyName' => $ipAddress, 'FriendlyName' => $ipAddress,
'Name' => $poolName, 'Name' => $poolName,
'InterfaceGroupID' => $config->{'interface_group'}, 'InterfaceGroupID' => $config->{'interface_group'},
'ClassID' => $trafficClassID, 'TrafficClassID' => $trafficClassID,
'TrafficLimitTx' => $trafficLimitTx, 'TxCIR' => $txCIR,
'TrafficLimitRx' => $trafficLimitRx, 'RxCIR' => $rxCIR,
'TrafficLimitTxBurst' => $trafficLimitTxBurst, 'TxLimit' => $txLimit,
'TrafficLimitRxBurst' => $trafficLimitRxBurst, 'RxLimit' => $rxLimit,
'Expires' => $now + $config->{'expiry_period'}, 'Expires' => $now + $config->{'expiry_period'},
'Source' => "plugin.radius", 'Source' => "plugin.radius",
}); });
...@@ -487,7 +534,12 @@ sub _session_socket_read ...@@ -487,7 +534,12 @@ sub _session_socket_read
my @txtChanges; my @txtChanges;
foreach my $item (keys %{$changes}) { foreach my $item (keys %{$changes}) {
push(@txtChanges,sprintf("%s = %s",$item,$changes->{$item})); # Make expires look nice
my $value = $changes->{$item};
if ($item eq "Expires") {
$value = sprintf("%s [%s]",$value,scalar(localtime($value)));
}
push(@txtChanges,sprintf("%s = %s",$item,$value));
} }
if (@txtChanges) { if (@txtChanges) {
$logger->log(LOG_INFO,"[RADIUS] Pool '%s' member '%s' updated: %s", $logger->log(LOG_INFO,"[RADIUS] Pool '%s' member '%s' updated: %s",
...@@ -497,6 +549,12 @@ sub _session_socket_read ...@@ -497,6 +549,12 @@ sub _session_socket_read
); );
} }
# TODO: Add output of updated items here too?
changePool({
'ID' => $pid,
'FriendlyName' => $ipAddress
});
# If not display message # If not display message
} else { } else {
...@@ -519,6 +577,12 @@ sub _session_socket_read ...@@ -519,6 +577,12 @@ sub _session_socket_read
'Expires' => $now + $config->{'expiry_period'}, 'Expires' => $now + $config->{'expiry_period'},
'Source' => "plugin.radius", 'Source' => "plugin.radius",
}); });
# TODO: Add output of updated items here too?
changePool({
'ID' => $pid,
'FriendlyName' => $ipAddress
});
} }
# Radius user going offline # Radius user going offline
...@@ -534,14 +598,20 @@ sub _session_socket_read ...@@ -534,14 +598,20 @@ sub _session_socket_read
# If there is only 1 pool member, then lets expire the pool in the removal expiry period # If there is only 1 pool member, then lets expire the pool in the removal expiry period
if (@poolMembers == 1) { if (@poolMembers == 1) {
$logger->log(LOG_INFO,"[RADIUS] Expiring pool '$poolName'"); $logger->log(LOG_INFO,"[RADIUS] Expiring pool '$poolName'");
changePool($pool->{'ID'},{ 'Expires' => $now + REMOVE_EXPIRY_PERIOD }); changePool({
'ID' => $pool->{'ID'},
'Expires' => $now + REMOVE_EXPIRY_PERIOD
});
} }
} }
# Check if we have a pool member with this username and IP # Check if we have a pool member with this username and IP
if (my $pmid = getPoolMemberByUsernameIP($pool->{'ID'},$username,$ipAddress)) { if (my $pmid = getPoolMemberByUsernameIP($pool->{'ID'},$username,$ipAddress)) {
$logger->log(LOG_INFO,"[RADIUS] Expiring pool '$poolName' member '$username'"); $logger->log(LOG_INFO,"[RADIUS] Expiring pool '$poolName' member '$username'");
changePoolMember($pmid,{ 'Expires' => $now + REMOVE_EXPIRY_PERIOD }); changePoolMember({
'ID' => $pmid,
'Expires' => $now + REMOVE_EXPIRY_PERIOD
});
} }
$logger->log(LOG_INFO,"[RADIUS] Pool '$poolName' member '$username' set to expire as they're offline"); $logger->log(LOG_INFO,"[RADIUS] Pool '$poolName' member '$username' set to expire as they're offline");
...@@ -559,6 +629,7 @@ sub _session_socket_read ...@@ -559,6 +629,7 @@ sub _session_socket_read
} }
# Convert status into something easy to useful # Convert status into something easy to useful
sub getStatus sub getStatus
{ {
...@@ -576,16 +647,17 @@ sub getStatus ...@@ -576,16 +647,17 @@ sub getStatus
} }
# Simple function to reduce everything to kbit # Simple function to reduce everything to kbit
sub getKbit sub getKbit
{ {
my ($counter,$quantifier) = @_; my ($counter,$quantifier) = @_;
# If there is no counter # If there is no counter
return undef if (!defined($counter)); return if (!defined($counter));
# We need a quantifier # We need a quantifier
return undef if (!defined($quantifier)); return if (!defined($quantifier));
# Initialize counter # Initialize counter
my $newCounter = $counter; my $newCounter = $counter;
...@@ -595,12 +667,13 @@ sub getKbit ...@@ -595,12 +667,13 @@ sub getKbit
} elsif ($quantifier =~ /^k$/i) { } elsif ($quantifier =~ /^k$/i) {
$newCounter = $counter * 1; $newCounter = $counter * 1;
} else { } else {
return undef; return;
} }
return $newCounter; return $newCounter;
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper Traffic shaping statistics # OpenTrafficShaper Traffic shaping statistics
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2015, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -21,16 +21,18 @@ package opentrafficshaper::plugins::statistics; ...@@ -21,16 +21,18 @@ package opentrafficshaper::plugins::statistics;
use strict; use strict;
use warnings; use warnings;
use DBI; use Data::Dumper;
use POE; use POE;
use Storable qw( dclone );
use awitpt::db::dblayer;
use opentrafficshaper::constants; use opentrafficshaper::constants;
use opentrafficshaper::logger; use opentrafficshaper::logger;
use opentrafficshaper::utils;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
getPool getPool
getPools getPools
getPoolMembers
getPoolTxInterface getPoolTxInterface
getPoolRxInterface getPoolRxInterface
...@@ -56,26 +58,110 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -56,26 +58,110 @@ our (@ISA,@EXPORT,@EXPORT_OK);
@EXPORT_OK = qw( @EXPORT_OK = qw(
getLastStats getLastStats
getStatsByClass getStatsBySID
getStatsByCounter getStatsBasicBySID
getSIDFromCID getSIDFromCID
getSIDFromLID getSIDFromPID
); );
use constant { use constant {
VERSION => '0.2.2', VERSION => '0.2.2',
# How often our config check ticks # How often our config check ticks
TICK_PERIOD => 5, TICK_PERIOD => 2,
STATISTICS_PERIOD => 60, STATISTICS_PERIOD => 5,
STATISTICS_DIR_TX => 1, STATISTICS_DIR_TX => 1,
STATISTICS_DIR_RX => 2, STATISTICS_DIR_RX => 2,
STATISTICS_MAXFLUSH_PER_PERIOD => 10000, STATISTICS_MAXFLUSH_PER_PERIOD => 10000,
# SQL Statements
SQL_ADD_IDENTIFIER => 'INSERT INTO identifiers (`Identifier`) VALUES (?)',
SQL_GET_IDENTIFIER => 'SELECT ID FROM identifiers WHERE `Identifier` = ?',
SQL_CONSOLIDATE_STATS => '
SELECT
`IdentifierID`, `Timestamp` - (`Timestamp` % ?) AS Timestamp,
`Direction`,
MAX(`CIR`) AS `CIR`, MAX(`Limit`) AS `Limit`, MAX(`Rate`) AS `Rate`, MAX(`PPS`) AS `PPS`,
MAX(`QueueLen`) AS `QueueLen`, MAX(`TotalBytes`) AS `TotalBytes`, MAX(`TotalPackets`) AS `TotalPackets`,
MAX(`TotalOverLimits`) AS `TotalOverLimits`, MAX(`TotalDropped`) AS `TotalDropped`
FROM
stats
WHERE
`Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
GROUP BY
`IdentifierID`, `Timestamp`, `Direction`
',
SQL_CONSOLIDATE_STATS_BASIC => '
SELECT
`IdentifierID`, `Timestamp` - (`Timestamp` % ?) AS Timestamp,
MAX(`Counter`) AS `Counter`
FROM
stats_basic
WHERE
`Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
GROUP BY
`IdentifierID`, `Timestamp`
',
SQL_GET_STATS => '
SELECT
`Timestamp`, `Direction`, `Rate`, `PPS`, `CIR`, `Limit`
FROM
stats
WHERE
`IdentifierID` = ?
AND `Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
',
SQL_GET_STATS_BASIC => '
SELECT
`Timestamp`, `Counter`
FROM
stats_basic
WHERE
`IdentifierID` = ?
AND `Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
',
SQL_CLEANUP_STATS => 'DELETE FROM stats WHERE `Key` = ? AND `Timestamp` < ?',
SQL_CLEANUP_STATS_BASIC => 'DELETE FROM stats_basic WHERE `Key` = ? AND `Timestamp` < ?'
}; };
sub STATS_CONFIG
{
{
1 => {
'precision' => 300, # 5min
'retention' => 4, # 4 days
},
2 => {
'precision' => 900, # 15min
'retention' => 14, # 14 days
},
3 => {
'precision' => 3600, # 1hr
'retention' => 28 * 2, # 2 months
},
4 => {
'precision' => 21600, # 6hr
'retention' => 28 * 6, # 6 months
},
5 => {
'precision' => 86400, # 24hr
'retention' => 28 * 12 * 2, # 2 years
}
}
}
# Plugin info # Plugin info
our $pluginInfo = { our $pluginInfo = {
...@@ -87,53 +173,34 @@ our $pluginInfo = { ...@@ -87,53 +173,34 @@ our $pluginInfo = {
}; };
# Copy of system globals # Copy of system globals
my $globals; my $globals;
my $logger; my $logger;
# Our configuration
my $config = {
'db_dsn' => undef,
'db_username' => "",
'db_password' => "",
};
# Stats configuration # Handle of DBI
my $statsConfig = { #
1 => { # $globals->{'Database'}->{'Handle'}
'precision' => 300, # $globals->{'Database'}->{'DSN'}
'retention' => 2, # 2 days # $globals->{'Database'}->{'Username'}
}, # $globals->{'Database'}->{'Password'}
2 => {
'precision' => 900,
'retention' => 14, # 2 week
},
3 => {
'precision' => 3600,
'retention' => 28 * 2, # 2 months
},
4 => {
'precision' => 21600, # 6hr
'retention' => 28 * 12 * 2, # 2 years
},
};
# DB identifier map
#
# $globals->{'IdentifierMap'}
# Handle of DBI
my $dbh;
# DB user mappings
my $statsDBIdentifierMap = { };
# Stats queue # Stats queue
my $statsQueue = [ ]; #
# Stats ubscribers # $globals->{'StatsQueue'}
my $subscribers; # $globals->{'LastCleanup'}
# Prepared statements we need... # $globals->{'LastConfigManagerStats'}
my $statsPreparedStatements = { };
# Last cleanup time # Stats subscribers & counter
my $lastCleanup = { }; # $globals->{'SIDSubscribers'}
# Last config manager stats pull # $globals->{'SSIDMap'}
my $lastConfigManagerStats = 0; # $globals->{'SSIDCounter'}
# $globals->{'SSIDCounterFreeList'}
# Initialize plugin # Initialize plugin
...@@ -147,21 +214,36 @@ sub plugin_init ...@@ -147,21 +214,36 @@ sub plugin_init
$logger->log(LOG_NOTICE,"[STATISTICS] OpenTrafficShaper Statistics v%s - Copyright (c) 2007-2014, AllWorldIT",VERSION); $logger->log(LOG_NOTICE,"[STATISTICS] OpenTrafficShaper Statistics v%s - Copyright (c) 2007-2014, AllWorldIT",VERSION);
# Initialize
$globals->{'Database'} = undef;
$globals->{'IdentifierMap'} = { };
$globals->{'StatsQueue'} = [ ];
$globals->{'LastCleanup'} = { };
$globals->{'LastConfigManagerStats'} = { };
$globals->{'SIDSubscribers'} = { };
$globals->{'SSIDMap'} = { };
$globals->{'SSIDCounter'} = 0;
$globals->{'SSIDCounterFreeList'} = [ ];
# Check our interfaces # Check our interfaces
if (defined(my $dbdsn = $globals->{'file.config'}->{'plugin.statistics'}->{'db_dsn'})) { if (defined(my $dbdsn = $globals->{'file.config'}->{'plugin.statistics'}->{'db_dsn'})) {
$logger->log(LOG_INFO,"[STATISTICS] Set db_dsn to '%s'",$dbdsn); $logger->log(LOG_INFO,"[STATISTICS] Set database DSN to '%s'",$dbdsn);
$config->{'db_dsn'} = $dbdsn; $globals->{'Database'}->{'DSN'} = $dbdsn;
if (defined(my $dbuser = $globals->{'file.config'}->{'plugin.statistics'}->{'db_username'})) {
$logger->log(LOG_INFO,"[STATISTICS] Set database username to '%s'",$dbuser);
$globals->{'Database'}->{'Username'} = $dbuser;
}
if (defined(my $dbpass = $globals->{'file.config'}->{'plugin.statistics'}->{'db_password'})) {
$logger->log(LOG_INFO,"[STATISTICS] Set database password to '%s'",$dbpass);
$globals->{'Database'}->{'Password'} = $dbpass;
}
} else { } else {
$logger->log(LOG_WARN,"[STATISTICS] No db_dsn to specified in configuration file. Stats storage disabled!"); $logger->log(LOG_WARN,"[STATISTICS] No database DSN to specified in configuration file. Stats storage disabled!");
}
if (defined(my $dbuser = $globals->{'file.config'}->{'plugin.statistics'}->{'db_username'})) {
$logger->log(LOG_INFO,"[STATISTICS] Set db_username to '%s'",$dbuser);
$config->{'db_username'} = $dbuser;
}
if (defined(my $dbpass = $globals->{'file.config'}->{'plugin.statistics'}->{'db_password'})) {
$logger->log(LOG_INFO,"[STATISTICS] Set db_password to '%s'",$dbpass);
$config->{'db_password'} = $dbpass;
} }
# This is our main stats session # This is our main stats session
...@@ -173,114 +255,43 @@ sub plugin_init ...@@ -173,114 +255,43 @@ sub plugin_init
# Stats update event # Stats update event
update => \&_session_update, update => \&_session_update,
# Subscription events
subscribe => \&_session_subscribe,
unsubscribe => \&_session_unsubscribe,
} }
); );
# Create DBI agent # Create DBI agent
if (defined($config->{'db_dsn'})) { if (defined($globals->{'Database'})) {
$dbh = DBI->connect( $globals->{'Database'}->{'Handle'} = DBInit($globals->{'Database'});
$config->{'db_dsn'}, $config->{'db_username'}, $config->{'db_password'}, # Check if handle is defined
{ if (defined($globals->{'Database'}->{'Handle'})) {
'AutoCommit' => 1, # Try connect (0 is success)
'RaiseError' => 0, if (!DBConnect()) {
'FetchHashKeyName' => 'NAME_lc' $logger->log(LOG_INFO,"[STATISTICS] Connected to database");
} } else {
); $logger->log(LOG_ERR,"[STATISTICS] Failed to connect to database: %s (DATABASE DISABLED)",
if (!defined($dbh)) { awitpt::db::dblayer::Error());
$logger->log(LOG_ERR,"[STATISTICS] Failed to connect to database: %s",$DBI::errstr); # Don't try again
} delete($globals->{'Database'});
}
# Prepare identifier add statement # If the handle is not defined, the database won't work
if ($dbh && (my $res = $dbh->prepare('INSERT INTO identifiers (`Identifier`) VALUES (?)'))) {
$statsPreparedStatements->{'identifier_add'} = $res;
} else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'identifier_add': %s",$DBI::errstr);
$dbh->disconnect();
$dbh = undef;
}
# Prepare identifier get statement
if ($dbh && (my $res = $dbh->prepare('SELECT ID FROM identifiers WHERE `Identifier` = ?'))) {
$statsPreparedStatements->{'identifier_get'} = $res;
} else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'identifier_get': %s",$DBI::errstr);
$dbh->disconnect();
$dbh = undef;
}
# Prepare stats consolidation statements
if ($dbh && (my $res = $dbh->prepare('
SELECT
`IdentifierID`, `Timestamp` - (`Timestamp` % ?) AS TimestampM,
`Direction`,
MAX(`CIR`) AS `CIR`, MAX(`Limit`) AS `Limit`, MAX(`Rate`) AS `Rate`, MAX(`PPS`) AS `PPS`,
MAX(`Queue_Len`) AS `Queue_Len`, AVG(`Total_Bytes`) AS `Total_Bytes`, AVG(`Total_Packets`) AS `Total_Packets`,
AVG(`Total_Overlimits`) AS `Total_Overlimits`, AVG(`Total_Dropped`) AS `Total_Dropped`
FROM
stats
WHERE
`Key` = ?
AND `Timestamp` < ?
GROUP BY
`IdentifierID`, `TimestampM`, `Direction`
'))) {
$statsPreparedStatements->{'stats_consolidate'} = $res;
} else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'stats_consolidate': %s",$DBI::errstr);
$dbh->disconnect();
$dbh = undef;
}
if ($dbh && (my $res = $dbh->prepare('
SELECT
`IdentifierID`, `Timestamp` - (`Timestamp` % ?) AS TimestampM,
MAX(`Counter`) AS `Counter`
FROM
stats_basic
WHERE
`Key` = ?
AND `Timestamp` < ?
GROUP BY
`IdentifierID`, `TimestampM`
'))) {
$statsPreparedStatements->{'stats_basic_consolidate'} = $res;
} else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'stats_basic_consolidate': %s",$DBI::errstr);
$dbh->disconnect();
$dbh = undef;
}
# Prepare stats cleanup statements
if ($dbh && (my $res = $dbh->prepare('DELETE FROM stats WHERE `Key` = ? AND `Timestamp` < ?'))) {
$statsPreparedStatements->{'stats_cleanup'} = $res;
} else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'stats_cleanup': %s",$DBI::errstr);
$dbh->disconnect();
$dbh = undef;
}
if ($dbh && (my $res = $dbh->prepare('DELETE FROM stats_basic WHERE `Key` = ? AND `Timestamp` < ?'))) {
$statsPreparedStatements->{'stats_basic_cleanup'} = $res;
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to prepare statement 'stats_basic_cleanup': %s",$DBI::errstr); $logger->log(LOG_ERR,"[STATISTICS] Failed to initailize database: %s (DATABASE DISABLED)",
$dbh->disconnect(); awitpt::db::dblayer::Error());
$dbh = undef;
} }
# Set last cleanup to now # Set last cleanup to now
my $now = time(); my $now = time();
foreach my $key (keys %{$statsConfig}) { foreach my $key (keys %{STATS_CONFIG()}) {
# Get aligned time so we cleanup sooner # Get aligned time so we cleanup sooner
$lastCleanup->{$key} = _getAlignedTime($now,$statsConfig->{$key}->{'precision'}); $globals->{'LastCleanup'}->{$key} = _getAlignedTime($now,STATS_CONFIG()->{$key}->{'precision'});
} }
$lastConfigManagerStats = $now; $globals->{'LastConfigManagerStats'} = $now;
} }
return 1; return 1;
} }
# Start the plugin # Start the plugin
sub plugin_start sub plugin_start
{ {
...@@ -289,6 +300,7 @@ sub plugin_start ...@@ -289,6 +300,7 @@ sub plugin_start
} }
# Initialize this plugins main POE session # Initialize this plugins main POE session
sub _session_start sub _session_start
{ {
...@@ -305,6 +317,7 @@ sub _session_start ...@@ -305,6 +317,7 @@ sub _session_start
} }
# Stop session # Stop session
sub _session_stop sub _session_stop
{ {
...@@ -316,13 +329,6 @@ sub _session_stop ...@@ -316,13 +329,6 @@ sub _session_stop
# Tear down data # Tear down data
$globals = undef; $globals = undef;
$dbh = undef;
$statsDBIdentifierMap = { };
$statsQueue = [ ];
$subscribers = undef;
$statsPreparedStatements = { };
$lastCleanup = { };
$lastConfigManagerStats = 0;
$logger->log(LOG_DEBUG,"[STATISTICS] Shutdown"); $logger->log(LOG_DEBUG,"[STATISTICS] Shutdown");
...@@ -330,28 +336,23 @@ sub _session_stop ...@@ -330,28 +336,23 @@ sub _session_stop
} }
# Time ticker for processing changes # Time ticker for processing changes
sub _session_tick sub _session_tick
{ {
my ($kernel,$heap) = @_[KERNEL,HEAP]; my ($kernel,$heap) = @_[KERNEL,HEAP];
# If we don't have a DB handle, just skip... # If we don't have a database, just skip...
if (!$dbh) { if (!$globals->{'Database'}) {
return; return;
} }
my $now = time(); my $now = time();
my $timer1 = [gettimeofday]; my $timer1 = [gettimeofday];
# Pull in statements
my $sthStatsConsolidate = $statsPreparedStatements->{'stats_consolidate'};
my $sthStatsCleanup = $statsPreparedStatements->{'stats_cleanup'};
my $sthStatsBasicConsolidate = $statsPreparedStatements->{'stats_basic_consolidate'};
my $sthStatsBasicCleanup = $statsPreparedStatements->{'stats_basic_cleanup'};
# Even out flushing over 10s to absorb spikes # Even out flushing over 10s to absorb spikes
my $totalFlush = @{$statsQueue}; my $totalFlush = @{$globals->{'StatsQueue'}};
my $maxFlush = int($totalFlush / 10) + 100; my $maxFlush = int($totalFlush / 10) + 100;
my $numFlush = 0; my $numFlush = 0;
...@@ -363,22 +364,22 @@ sub _session_tick ...@@ -363,22 +364,22 @@ sub _session_tick
# Loop and build the data to create our multi-insert # Loop and build the data to create our multi-insert
my (@insertHolders,@insertBasicHolders); my (@insertHolders,@insertBasicHolders);
my (@insertData,@insertBasicData); my (@insertData,@insertBasicData);
while (defined(my $stat = shift(@{$statsQueue})) && $numFlush < $maxFlush) { while (defined(my $stat = shift(@{$globals->{'StatsQueue'}})) && $numFlush < $maxFlush) {
# This is a basic counter # This is a basic counter
if (defined($stat->{'counter'})) { if (defined($stat->{'Counter'})) {
push(@insertBasicHolders,"(?,?,?,?)"); push(@insertBasicHolders,"(?,?,?,?)");
push(@insertBasicData, push(@insertBasicData,
$stat->{'identifierid'}, $stat->{'key'}, $stat->{'timestamp'}, $stat->{'IdentifierID'}, $stat->{'Key'}, $stat->{'Timestamp'},
$stat->{'counter'} $stat->{'Counter'}
); );
# Full stats counter # Full stats counter
} else { } else {
push(@insertHolders,"(?,?,?,?,?,?,?,?,?,?,?,?,?)"); push(@insertHolders,"(?,?,?,?,?,?,?,?,?,?,?,?,?)");
push(@insertData, push(@insertData,
$stat->{'identifierid'}, $stat->{'key'}, $stat->{'timestamp'}, $stat->{'IdentifierID'}, $stat->{'Key'}, $stat->{'Timestamp'},
$stat->{'direction'}, $stat->{'Direction'},
$stat->{'cir'}, $stat->{'limit'}, $stat->{'rate'}, $stat->{'pps'}, $stat->{'queue_len'}, $stat->{'CIR'}, $stat->{'Limit'}, $stat->{'Rate'}, $stat->{'PPS'}, $stat->{'QueueLen'},
$stat->{'total_bytes'}, $stat->{'total_packets'}, $stat->{'total_overlimits'}, $stat->{'total_dropped'} $stat->{'TotalBytes'}, $stat->{'TotalPackets'}, $stat->{'TotalOverLimits'}, $stat->{'TotalDropped'}
); );
} }
...@@ -387,36 +388,36 @@ sub _session_tick ...@@ -387,36 +388,36 @@ sub _session_tick
# If we got things to insert, do it # If we got things to insert, do it
if (@insertBasicHolders > 0) { if (@insertBasicHolders > 0) {
my $res = $dbh->do(' my $res = DBDo('
INSERT DELAYED INTO stats_basic INSERT INTO stats_basic
( (
`IdentifierID`, `Key`, `Timestamp`, `IdentifierID`, `Key`, `Timestamp`,
`Counter` `Counter`
) )
VALUES VALUES
'.join(',',@insertBasicHolders),undef,@insertBasicData '.join(',',@insertBasicHolders),@insertBasicData
); );
# Check for error # Check for error
if (!defined($res)) { if (!defined($res)) {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute delayed stats_basic insert: %s",$DBI::errstr); $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic insert: %s",awitpt::db::dblayer::Error());
} }
} }
# And normal stats... # And normal stats...
if (@insertHolders > 0) { if (@insertHolders > 0) {
my $res = $dbh->do(' my $res = DBDo('
INSERT DELAYED INTO stats INSERT INTO stats
( (
`IdentifierID`, `Key`, `Timestamp`, `IdentifierID`, `Key`, `Timestamp`,
`Direction`, `Direction`,
`CIR`, `Limit`, `Rate`, `PPS`, `Queue_Len`, `CIR`, `Limit`, `Rate`, `PPS`, `QueueLen`,
`Total_Bytes`, `Total_Packets`, `Total_Overlimits`, `Total_Dropped` `TotalBytes`, `TotalPackets`, `TotalOverLimits`, `TotalDropped`
) )
VALUES VALUES
'.join(',',@insertHolders),undef,@insertData '.join(',',@insertHolders),@insertData
); );
# Check for error # Check for error
if (!defined($res)) { if (!defined($res)) {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute delayed stats insert: %s",$DBI::errstr); $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats insert: %s",awitpt::db::dblayer::Error());
} }
} }
...@@ -425,24 +426,24 @@ sub _session_tick ...@@ -425,24 +426,24 @@ sub _session_tick
if ($numFlush) { if ($numFlush) {
my $timediff2 = tv_interval($timer1,$timer2); my $timediff2 = tv_interval($timer1,$timer2);
$logger->log(LOG_INFO,"[STATISTICS] Total stats flush time %s/%s records: %s", $logger->log(LOG_INFO,"[STATISTICS] Total stats flush time %s/%s records: %s",
$numFlush, $numFlush,
$totalFlush, $totalFlush,
sprintf('%.3fs',$timediff2) sprintf('%.3fs',$timediff2)
); );
} }
my $res; my $res;
# Loop with our stats consolidation configuration # Loop with our stats consolidation configuration
foreach my $key (sort keys %{$statsConfig}) { foreach my $key (sort keys %{STATS_CONFIG()}) {
my $timerA = [gettimeofday]; my $timerA = [gettimeofday];
my $precision = $statsConfig->{$key}->{'precision'}; my $precision = STATS_CONFIG()->{$key}->{'precision'};
my $thisPeriod = _getAlignedTime($now,$precision); my $thisPeriod = _getAlignedTime($now,$precision);
my $lastPeriod = $thisPeriod - $precision; my $lastPeriod = $thisPeriod - $precision;
my $prevKey = $key - 1; my $prevKey = $key - 1;
# If we havn't exited the last period, then skip # If we havn't exited the last period, then skip
if ($lastCleanup->{$key} > $lastPeriod) { if ($globals->{'LastCleanup'}->{$key} > $lastPeriod) {
next; next;
} }
...@@ -450,60 +451,65 @@ sub _session_tick ...@@ -450,60 +451,65 @@ sub _session_tick
my $numStatsBasicConsolidated = 0; my $numStatsBasicConsolidated = 0;
my $numStatsConsolidated = 0; my $numStatsConsolidated = 0;
my $consolidateFrom = $lastPeriod - $precision * 2;
my $consolidateUpTo = $lastPeriod - $precision; my $consolidateUpTo = $lastPeriod - $precision;
# Execute and pull in consolidated stats # Execute and pull in consolidated stats
$res = $sthStatsBasicConsolidate->execute($precision,$prevKey,$consolidateUpTo); $res = DBSelect(SQL_CONSOLIDATE_STATS_BASIC,$precision,$prevKey,$consolidateFrom,$consolidateUpTo);
if ($res) { if ($res) {
# Loop with items returned # Loop with items returned
while (my $item = $sthStatsBasicConsolidate->fetchrow_hashref()) { while (my $item = hashifyLCtoMC($res->fetchrow_hashref(),'IdentifierID','Timestamp','Counter')) {
$item->{'key'} = $key; $item->{'Key'} = $key;
$item->{'timestamp'} = $item->{'timestampm'};
# Queue for insert # Queue for insert
push(@{$statsQueue},$item); push(@{$globals->{'StatsQueue'}},$item);
$numStatsBasicConsolidated++; $numStatsBasicConsolidated++;
} }
DBFreeRes($res);
# If there was an error, make sure we report it # If there was an error, make sure we report it
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic consolidation statement: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic consolidation statement: %s",
$sthStatsBasicConsolidate->errstr() awitpt::db::dblayer::Error());
);
} }
# And the normal stats... # And the normal stats...
$res = $sthStatsConsolidate->execute($precision,$prevKey,$consolidateUpTo); $res = DBSelect(SQL_CONSOLIDATE_STATS,$precision,$prevKey,$consolidateFrom,$consolidateUpTo);
if ($res) { if ($res) {
# Loop with items returned # Loop with items returned
while (my $item = $sthStatsConsolidate->fetchrow_hashref()) { while (my $item = hashifyLCtoMC(
$item->{'key'} = $key; $res->fetchrow_hashref(),
$item->{'timestamp'} = $item->{'timestampm'}; 'IdentifierID','Timestamp','Direction','CIR','Limit','Rate','PPS','QueueLen','TotalBytes','TotalPackets',
'TotalOverLimits','TotalDropped'
)) {
$item->{'Key'} = $key;
# Queue for insert # Queue for insert
push(@{$statsQueue},$item); push(@{$globals->{'StatsQueue'}},$item);
$numStatsConsolidated++; $numStatsConsolidated++;
} }
DBFreeRes($res);
# If there was an error, make sure we report it # If there was an error, make sure we report it
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats consolidation statement: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats consolidation statement: %s",
$sthStatsConsolidate->errstr() awitpt::db::dblayer::Error());
);
} }
# Set last cleanup to now # Set last cleanup to now
$lastCleanup->{$key} = $now; $globals->{'LastCleanup'}->{$key} = $now;
my $timerB = [gettimeofday]; my $timerB = [gettimeofday];
my $timediffB = tv_interval($timerA,$timerB); my $timediffB = tv_interval($timerA,$timerB);
$logger->log(LOG_INFO,"[STATISTICS] Stats consolidation time for key %s: %s (%s basic, %s normal), up to %s [%s]", $logger->log(LOG_INFO,"[STATISTICS] Stats consolidation: key %s in %s (%s basic, %s normal), period %s - %s [%s - %s]",
$key, $key,
sprintf('%.3fs',$timediffB), sprintf('%.3fs',$timediffB),
$numStatsBasicConsolidated, $numStatsBasicConsolidated,
$numStatsConsolidated, $numStatsConsolidated,
$consolidateUpTo, $consolidateFrom,
scalar(localtime($consolidateUpTo)) $consolidateUpTo,
scalar(localtime($consolidateFrom)),
scalar(localtime($consolidateUpTo))
); );
} }
...@@ -513,96 +519,116 @@ sub _session_tick ...@@ -513,96 +519,116 @@ sub _session_tick
# We only need to run as often as the first precision # We only need to run as often as the first precision
# - If cleanup has not yet run? # - If cleanup has not yet run?
# - or if the 0 cleanup plus precision of the first key is in the past (data is now stale?) # - or if the 0 cleanup plus precision of the first key is in the past (data is now stale?)
if (!defined($lastCleanup->{'0'}) || $lastCleanup->{'0'} + $statsConfig->{1}->{'precision'} < $now) { if (!defined($globals->{'LastCleanup'}->{'0'}) || $globals->{'LastCleanup'}->{'0'} + STATS_CONFIG()->{1}->{'precision'} < $now) {
# We're going to clean up for the first stats precision * 3, which should be enough # We're going to clean up for the first stats precision * 3, which should be enough
my $cleanUpTo = $now - ($statsConfig->{1}->{'precision'} * 3); my $cleanUpTo = $now - (STATS_CONFIG()->{1}->{'precision'} * 3);
# Streamed stats is removed 3 time periods past the first precision # Streamed stats is removed 3 time periods past the first precision
if ($res = $sthStatsBasicCleanup->execute(0, $cleanUpTo)) { my $timerA = [gettimeofday];
if ($res = DBDo(SQL_CLEANUP_STATS_BASIC,0,$cleanUpTo)) {
my $timerB = [gettimeofday];
my $timerdiffA = tv_interval($timerA,$timerB);
# We get 0E0 for 0 when none were removed # We get 0E0 for 0 when none were removed
if ($res ne "0E0") { if ($res ne "0E0") {
$logger->log(LOG_INFO,"[STATISTICS] Cleanup streamed stats_basic %s, up to %s [%s]", $logger->log(LOG_INFO,"[STATISTICS] Cleanup streamed stats_basic, %s items in %s, up to %s [%s]",
$res, $res,
$cleanUpTo, sprintf('%.3fs',$timerdiffA),
scalar(localtime($cleanUpTo)) $cleanUpTo,
scalar(localtime($cleanUpTo)),
); );
} }
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic cleanup statement: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic cleanup statement: %s",
$sthStatsBasicCleanup->errstr() awitpt::db::dblayer::Error());
);
} }
# And the normal stats... # And the normal stats...
if ($res = $sthStatsCleanup->execute(0, $cleanUpTo)) { $timerA = [gettimeofday];
if ($res = DBDo(SQL_CLEANUP_STATS,0,$cleanUpTo)) {
my $timerB = [gettimeofday];
my $timerdiffA = tv_interval($timerA,$timerB);
# We get 0E0 for 0 when none were removed # We get 0E0 for 0 when none were removed
if ($res ne "0E0") { if ($res ne "0E0") {
$logger->log(LOG_INFO,"[STATISTICS] Cleanup streamed stats %s, up to %s [%s]", $logger->log(LOG_INFO,"[STATISTICS] Cleanup streamed stats, %s items in %s, up to %s [%s]",
$res, $res,
$cleanUpTo,scalar(localtime($cleanUpTo)) sprintf('%.3fs',$timerdiffA),
$cleanUpTo,scalar(localtime($cleanUpTo))
); );
} }
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats cleanup statement: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats cleanup statement: %s",
$sthStatsCleanup->errstr() awitpt::db::dblayer::Error()
); );
} }
# Loop and remove retained stats # Loop and remove retained stats
foreach my $key (keys %{$statsConfig}) { foreach my $key (keys %{STATS_CONFIG()}) {
# Work out timestamp to clean up to by multiplying the retention period by days # Work out timestamp to clean up to by multiplying the retention period by days
$cleanUpTo = $now - ($statsConfig->{$key}->{'retention'} * 86400); $cleanUpTo = $now - (STATS_CONFIG()->{$key}->{'retention'} * 86400);
# Retention period is in # days # Retention period is in # days
if ($res = $sthStatsBasicCleanup->execute($key, $cleanUpTo)) { my $timerA = [gettimeofday];
if ($res = DBDo(SQL_CLEANUP_STATS_BASIC,$key,$cleanUpTo)) {
# We get 0E0 for 0 when none were removed # We get 0E0 for 0 when none were removed
if ($res ne "0E0") { if ($res ne "0E0") {
$logger->log(LOG_INFO,"[STATISTICS] Cleanup key %s stats_basic, %s items up to %s [%s]", my $timerB = [gettimeofday];
$key, my $timerdiffA = tv_interval($timerA,$timerB);
$res,
$cleanUpTo, $logger->log(LOG_INFO,"[STATISTICS] Cleanup stats_basic key %s in %s, %s items up to %s [%s]",
scalar(localtime($cleanUpTo)) $key,
sprintf('%.3fs',$timerdiffA),
$res,
$cleanUpTo,
scalar(localtime($cleanUpTo))
); );
} }
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic cleanup statement for key %s: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats_basic cleanup statement for key %s: %s",
$key, $key,
$sthStatsBasicCleanup->errstr() awitpt::db::dblayer::Error()
); );
} }
# And normal stats... # And normal stats...
if ($res = $sthStatsCleanup->execute($key, $cleanUpTo)) { $timerA = [gettimeofday];
if ($res = DBDo(SQL_CLEANUP_STATS,$key,$cleanUpTo)) {
# We get 0E0 for 0 when none were removed # We get 0E0 for 0 when none were removed
if ($res ne "0E0") { if ($res ne "0E0") {
$logger->log(LOG_INFO,"[STATISTICS] Cleanup key %s stats %s, up to %s [%s]", my $timerB = [gettimeofday];
$key, my $timerdiffA = tv_interval($timerA,$timerB);
$res,
$cleanUpTo, $logger->log(LOG_INFO,"[STATISTICS] Cleanup stats key %s in %s, %s items up to %s [%s]",
scalar(localtime($cleanUpTo)) $key,
sprintf('%.3fs',$timerdiffA),
$res,
$cleanUpTo,
scalar(localtime($cleanUpTo))
); );
} }
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats cleanup statement for key %s: %s", $logger->log(LOG_ERR,"[STATISTICS] Failed to execute stats cleanup statement for key %s: %s",
$key, $key,
$sthStatsCleanup->errstr() awitpt::db::dblayer::Error()
); );
} }
} }
# Set last main cleanup to now # Set last main cleanup to now
$lastCleanup->{'0'} = $now; $globals->{'LastCleanup'}->{'0'} = $now;
my $timer4 = [gettimeofday]; my $timer4 = [gettimeofday];
my $timediff4 = tv_interval($timer3,$timer4); my $timediff4 = tv_interval($timer3,$timer4);
$logger->log(LOG_INFO,"[STATISTICS] Stats cleanup time: %s", $logger->log(LOG_INFO,"[STATISTICS] Total stats cleanup time: %s",
sprintf('%.3fs',$timediff4) sprintf('%.3fs',$timediff4)
); );
} }
# Check if we need to pull config manager stats # Check if we need to pull config manager stats
if ($now - $lastConfigManagerStats > STATISTICS_PERIOD) { if ($now - $globals->{'LastConfigManagerStats'} > STATISTICS_PERIOD) {
my $configManagerStats = _getConfigManagerStats(); my $configManagerStats = _getConfigManagerStats();
_processStatistics($kernel,$configManagerStats); _processStatistics($kernel,$configManagerStats);
$lastConfigManagerStats = $now; $globals->{'LastConfigManagerStats'} = $now;
} }
# Set delay on config updates # Set delay on config updates
...@@ -610,6 +636,7 @@ sub _session_tick ...@@ -610,6 +636,7 @@ sub _session_tick
} }
# Update limit Statistics # Update limit Statistics
# $item has some special use cases: # $item has some special use cases:
# main:$iface:all - Interface total stats # main:$iface:all - Interface total stats
...@@ -620,39 +647,78 @@ sub _session_update ...@@ -620,39 +647,78 @@ sub _session_update
my ($kernel, $statsData) = @_[KERNEL, ARG0]; my ($kernel, $statsData) = @_[KERNEL, ARG0];
# TODO? This requires DB access
if (!$dbh) {
return;
}
_processStatistics($kernel,$statsData); _processStatistics($kernel,$statsData);
} }
# Handle subscriptions to updates # Handle subscriptions to updates
sub _session_subscribe sub subscribe
{ {
my ($kernel, $handler, $handlerEvent, $item) = @_[KERNEL, ARG0, ARG1, ARG2]; my ($sid,$conversions,$handler,$event) = @_;
$logger->log(LOG_INFO,"[STATISTICS] Got subscription request from '%s' for '%s' via event '%s'",$handler,$item,$handlerEvent); $logger->log(LOG_INFO,"[STATISTICS] Got subscription request for '%s': handler='%s', event='%s'",
$sid,
$handler,
$event
);
# Grab next SSID
my $ssid = shift(@{$globals->{'SSIDCounterFreeList'}});
if (!defined($ssid)) {
$ssid = $globals->{'SSIDCounter'}++;
}
$subscribers->{$item}->{$handler}->{$handlerEvent} = $item; # Setup data and conversions
$globals->{'SSIDMap'}->{$ssid} = $globals->{'SIDSubscribers'}->{$sid}->{$ssid} = {
'SID' => $sid,
'SSID' => $ssid,
'Conversions' => $conversions,
'Handler' => $handler,
'Event' => $event
};
# Return the SID we subscribed
return $ssid;
} }
# Handle unsubscribes # Handle unsubscribes
sub _session_unsubscribe sub unsubscribe
{ {
my ($kernel, $handler, $handlerEvent, $item) = @_[KERNEL, ARG0, ARG1, ARG2]; my $ssid = shift;
$logger->log(LOG_INFO,"[STATISTICS] Got unsubscription request for '%s' regarding '%s'",$handler,$item); # Grab item, and check if it doesnt exist
my $item = $globals->{'SSIDMap'}->{$ssid};
if (!defined($item)) {
$logger->log(LOG_ERR,"[STATISTICS] Got unsubscription request for SSID '%s' that doesn't exist",
$ssid
);
return
}
$logger->log(LOG_INFO,"[STATISTICS] Got unsubscription request for SSID '%s'",
$ssid
);
delete($subscribers->{$item}->{$handler}->{$handlerEvent}); # Remove subscriber
delete($globals->{'SIDSubscribers'}->{$item->{'SID'}}->{$ssid});
# If SID is now empty, remove it too
if (! keys %{$globals->{'SIDSubscribers'}->{$item->{'SID'}}}) {
delete($globals->{'SIDSubscribers'}->{$item->{'SID'}});
}
# Remove mapping
delete($globals->{'SSIDMap'}->{$ssid});
# Push onto list of free ID's
push(@{$globals->{'SSIDCounterFreeList'}},$ssid);
} }
# Return user last stats # Return user last stats
sub getLastStats sub getLastStats
{ {
...@@ -684,24 +750,53 @@ sub getLastStats ...@@ -684,24 +750,53 @@ sub getLastStats
} }
# Return stats by SID # Return stats by SID
sub getStatsBySID sub getStatsBySID
{ {
my $sid = shift; my ($sid,$conversions,$startTimestamp,$endTimestamp) = @_;
return _getStatsBySID($sid);
my $statistics = _getStatsBySID($sid,$startTimestamp,$endTimestamp);
if (!defined($statistics)) {
return;
}
# Loop and convert
foreach my $timestamp (keys %{$statistics}) {
my $stat = $statistics->{$timestamp};
# Use new item
$statistics->{$timestamp} = _fixStatDirection($stat,$conversions);
}
return $statistics;
} }
# Return basic stats by SID # Return basic stats by SID
sub getStatsBasicBySID sub getStatsBasicBySID
{ {
my $sid = shift; my ($sid,$conversions) = @_;
return _getStatsBasicBySID($sid);
my $statistics = _getStatsBasicBySID($sid);
if (!defined($statistics)) {
return;
}
# Loop and convert
foreach my $timestamp (keys %{$statistics}) {
my $stat = $statistics->{$timestamp};
# Use new item
$statistics->{$timestamp} = _fixCounterName($stat,$conversions);
}
return $statistics;
} }
# Get the stats ID from Class ID # Get the stats ID from Class ID
sub getSIDFromCID sub getSIDFromCID
{ {
...@@ -711,7 +806,7 @@ sub getSIDFromCID ...@@ -711,7 +806,7 @@ sub getSIDFromCID
# Grab identifier based on class ID # Grab identifier based on class ID
my $identifier = _getIdentifierFromCID($iface,$cid); my $identifier = _getIdentifierFromCID($iface,$cid);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# Return the SID fo the identifier # Return the SID fo the identifier
...@@ -719,6 +814,7 @@ sub getSIDFromCID ...@@ -719,6 +814,7 @@ sub getSIDFromCID
} }
# Set the stats ID from Class ID # Set the stats ID from Class ID
sub setSIDFromCID sub setSIDFromCID
{ {
...@@ -731,7 +827,7 @@ sub setSIDFromCID ...@@ -731,7 +827,7 @@ sub setSIDFromCID
# If not, grab the identifier # If not, grab the identifier
my $identifier = _getIdentifierFromCID($iface,$cid); my $identifier = _getIdentifierFromCID($iface,$cid);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# And setup a new SID # And setup a new SID
$sid = _setSIDFromIdentifier($identifier); $sid = _setSIDFromIdentifier($identifier);
...@@ -741,6 +837,7 @@ sub setSIDFromCID ...@@ -741,6 +837,7 @@ sub setSIDFromCID
} }
# Get the stats ID from a PID # Get the stats ID from a PID
sub getSIDFromPID sub getSIDFromPID
{ {
...@@ -750,7 +847,7 @@ sub getSIDFromPID ...@@ -750,7 +847,7 @@ sub getSIDFromPID
# Grab identifier from a PID # Grab identifier from a PID
my $identifier = _getIdentifierFromPID($pid); my $identifier = _getIdentifierFromPID($pid);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# Return the SID for the PID # Return the SID for the PID
...@@ -770,7 +867,7 @@ sub setSIDFromPID ...@@ -770,7 +867,7 @@ sub setSIDFromPID
# If we can't, grab the identifier instead # If we can't, grab the identifier instead
my $identifier = _getIdentifierFromPID($pid); my $identifier = _getIdentifierFromPID($pid);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# And setup the SID # And setup the SID
$sid = _setSIDFromIdentifier($identifier); $sid = _setSIDFromIdentifier($identifier);
...@@ -780,6 +877,7 @@ sub setSIDFromPID ...@@ -780,6 +877,7 @@ sub setSIDFromPID
} }
# Get the stats ID from a counter # Get the stats ID from a counter
sub getSIDFromCounter sub getSIDFromCounter
{ {
...@@ -789,7 +887,7 @@ sub getSIDFromCounter ...@@ -789,7 +887,7 @@ sub getSIDFromCounter
# Grab identifier from a counter # Grab identifier from a counter
my $identifier = _getIdentifierFromCounter($counter); my $identifier = _getIdentifierFromCounter($counter);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# Return the SID for the counter # Return the SID for the counter
...@@ -797,6 +895,7 @@ sub getSIDFromCounter ...@@ -797,6 +895,7 @@ sub getSIDFromCounter
} }
# Set the stats ID from a counter # Set the stats ID from a counter
sub setSIDFromCounter sub setSIDFromCounter
{ {
...@@ -809,7 +908,7 @@ sub setSIDFromCounter ...@@ -809,7 +908,7 @@ sub setSIDFromCounter
# If we can't, grab the identifier instead # If we can't, grab the identifier instead
my $identifier = _getIdentifierFromCounter($counter); my $identifier = _getIdentifierFromCounter($counter);
if (!defined($identifier)) { if (!defined($identifier)) {
return undef; return;
} }
# And setup the SID # And setup the SID
$sid = _setSIDFromIdentifier($identifier); $sid = _setSIDFromIdentifier($identifier);
...@@ -819,6 +918,7 @@ sub setSIDFromCounter ...@@ -819,6 +918,7 @@ sub setSIDFromCounter
} }
# Return traffic direction # Return traffic direction
sub getTrafficDirection sub getTrafficDirection
{ {
...@@ -836,31 +936,44 @@ sub getTrafficDirection ...@@ -836,31 +936,44 @@ sub getTrafficDirection
return STATISTICS_DIR_RX; return STATISTICS_DIR_RX;
} }
return undef; return;
} }
# Generate ConfigManager counters # Generate ConfigManager counters
sub getConfigManagerCounters sub getConfigManagerCounters
{ {
my @poolList = getPools(); my @poolList = getPools();
my $classes = getAllTrafficClasses(); my @classes = getAllTrafficClasses();
# Grab user count # Grab user count
my %counters; my %counters;
$counters{"ConfigManager:TotalPools"} = @poolList; $counters{"configmanager.totalpools"} = @poolList;
# Zero this counter
$counters{"configmanager.totalpoolmembers"} = 0;
# Zero the number of pools in each class to start off with # Zero the number of pools in each class to start off with
foreach my $cid (keys %{$classes}) { foreach my $cid (@classes) {
$counters{"ConfigManager:ClassPools:$cid"} = 0; $counters{"configmanager.classpools.$cid"} = 0;
$counters{"configmanager.classpoolmembers.$cid"} = 0;
} }
# Pull in each pool and bump up the class counter # Pull in each pool and bump up the class counter
foreach my $pid (@poolList) { foreach my $pid (@poolList) {
my $pool = getPool($pid);
my $cid = getPoolTrafficClassID($pid); my $cid = getPoolTrafficClassID($pid);
# Bump the class counter my @poolMembers = getPoolMembers($pid);
$counters{"ConfigManager:ClassLimits:$cid"}++; # Bump the class counters
$counters{"configmanager.classpools.$cid"}++;
$counters{"configmanager.classpoolmembers.$cid"} += @poolMembers;
# Bump the pool member counter
$counters{"configmanager.totalpoolmembers"} += @poolMembers;
# Set pool member count
$counters{"configmanager.poolmembers.$pool->{'InterfaceGroupID'}/$pool->{'Name'}"} = @poolMembers;
} }
return \%counters; return \%counters;
...@@ -871,41 +984,76 @@ sub getConfigManagerCounters ...@@ -871,41 +984,76 @@ sub getConfigManagerCounters
# Internal Functions # Internal Functions
# #
# Function to process a bunch of statistics # Function to process a bunch of statistics
sub _processStatistics sub _processStatistics
{ {
my ($kernel,$statsData) = @_; my ($kernel,$statsData) = @_;
my $queuedEvents;
# Loop through stats data we got # Loop through stats data we got
while ((my $sid, my $stat) = each(%{$statsData})) { while ((my $sid, my $stat) = each(%{$statsData})) {
$stat->{'identifierid'} = $sid; $stat->{'IdentifierID'} = $sid;
$stat->{'key'} = 0; $stat->{'Key'} = 0;
push(@{$statsQueue},$stat); # Add to main queue
# # Check if we have an event handler subscriber for this item push(@{$globals->{'StatsQueue'}},$stat);
# if (defined($subscribers->{$statsItem}) && %{$subscribers->{$statsItem}}) {
# # If we do, loop with them # Check if we have an event handler subscriber for this item
# foreach my $handler (keys %{$subscribers->{$statsItem}}) { if (defined(my $subscribers = $globals->{'SIDSubscribers'}->{$sid})) {
#
# # If no events are linked to this handler, continue # Build the stat that our conversions understands
# if (!(keys %{$subscribers->{$statsItem}->{$handler}})) { my $eventStat;
# next; # This is a basic counter
# } if (defined($stat->{'Counter'})) {
# $eventStat = {
# # Or ... If we have events, process them 'counter' => $stat->{'Counter'}
# foreach my $event (keys %{$subscribers->{$statsItem}->{$handler}}) { };
# } else {
# $kernel->post($handler => $event => $statsItem => $stat); $eventStat->{$stat->{'Direction'}} = {
# } 'rate' => $stat->{'Rate'},
# } 'pps' => $stat->{'PPS'},
# } 'cir' => $stat->{'CIR'},
'limit' => $stat->{'Limit'}
};
}
# If we do, loop with them
foreach my $ssid (keys %{$subscribers}) {
my $subscriber = $subscribers->{$ssid};
my $handler = $subscriber->{'Handler'};
my $event = $subscriber->{'Event'};
my $conversions = $subscriber->{'Conversions'};
# Get temp stat, this still refs the original one
my $tempStat;
# This is a basic counter
if (defined($eventStat->{'counter'})) {
$tempStat = _fixCounterName($eventStat,$conversions);
} else {
$tempStat = _fixStatDirection($eventStat,$conversions);
}
# Send a copy! so we don't send refs to data used elsewhere
$queuedEvents->{$handler}->{$event}->{$ssid}->{$stat->{'Timestamp'}} = dclone($tempStat);
}
}
}
# Loop with events we need to dispatch
foreach my $handler (keys %{$queuedEvents}) {
my $events = $queuedEvents->{$handler};
foreach my $event (keys %{$events}) {
$kernel->post($handler => $event => $queuedEvents->{$handler}->{$event});
}
} }
} }
# Generate ConfigManager stats # Generate ConfigManager stats
sub _getConfigManagerStats sub _getConfigManagerStats
{ {
...@@ -919,9 +1067,9 @@ sub _getConfigManagerStats ...@@ -919,9 +1067,9 @@ sub _getConfigManagerStats
foreach my $item (keys %{$counters}) { foreach my $item (keys %{$counters}) {
my $identifierID = setSIDFromCounter($item); my $identifierID = setSIDFromCounter($item);
my $stat = { my $stat = {
'identifierid' => $identifierID, 'IdentifierID' => $identifierID,
'timestamp' => $now, 'Timestamp' => $now,
'counter' => $counters->{$item} 'Counter' => $counters->{$item}
}; };
$statsData->{$identifierID} = $stat; $statsData->{$identifierID} = $stat;
} }
...@@ -930,6 +1078,7 @@ sub _getConfigManagerStats ...@@ -930,6 +1078,7 @@ sub _getConfigManagerStats
} }
# Function to get a SID identifier from a class ID # Function to get a SID identifier from a class ID
sub _getIdentifierFromCID sub _getIdentifierFromCID
{ {
...@@ -940,6 +1089,7 @@ sub _getIdentifierFromCID ...@@ -940,6 +1089,7 @@ sub _getIdentifierFromCID
} }
# Function to get a SID identifier from a pool ID # Function to get a SID identifier from a pool ID
sub _getIdentifierFromPID sub _getIdentifierFromPID
{ {
...@@ -948,13 +1098,14 @@ sub _getIdentifierFromPID ...@@ -948,13 +1098,14 @@ sub _getIdentifierFromPID
my $pool = getPool($pid); my $pool = getPool($pid);
if (!defined($pool)) { if (!defined($pool)) {
return undef; return;
} }
return sprintf("Pool:%s",$pool->{'Name'}); return sprintf("Pool:%s/%s",$pool->{'InterfaceGroupID'},$pool->{'Name'});
} }
# Function to get a SID identifier from a counter # Function to get a SID identifier from a counter
sub _getIdentifierFromCounter sub _getIdentifierFromCounter
{ {
...@@ -964,16 +1115,18 @@ sub _getIdentifierFromCounter ...@@ -964,16 +1115,18 @@ sub _getIdentifierFromCounter
} }
# Return a cached SID if its cached # Return a cached SID if its cached
sub _getCachedSIDFromIdentifier sub _getCachedSIDFromIdentifier
{ {
my $identifier = shift; my $identifier = shift;
return $statsDBIdentifierMap->{$identifier}; return $globals->{'IdentifierMap'}->{$identifier};
} }
# Grab or add the identifier to the DB # Grab or add the identifier to the DB
sub _getSIDFromIdentifier sub _getSIDFromIdentifier
{ {
...@@ -985,39 +1138,51 @@ sub _getSIDFromIdentifier ...@@ -985,39 +1138,51 @@ sub _getSIDFromIdentifier
return $sid; return $sid;
} }
# We need the DB to be alive to do this...
if (!defined($globals->{'Database'})) {
return;
}
# Try grab it from DB # Try grab it from DB
my $identifierGetSTH = $statsPreparedStatements->{'identifier_get'}; if (my $res = DBSelect(SQL_GET_IDENTIFIER,$identifier)) {
if (my $res = $identifierGetSTH->execute($identifier)) {
# Grab first row and return # Grab first row and return
if (my $row = $identifierGetSTH->fetchrow_hashref()) { if (my $row = $res->fetchrow_hashref()) {
return $statsDBIdentifierMap->{$identifier} = $row->{'id'}; DBFreeRes($res);
return $globals->{'IdentifierMap'}->{$identifier} = $row->{'id'};
} }
DBFreeRes($res);
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to get SID from identifier '%s': %s",$identifier,$identifierGetSTH->errstr); $logger->log(LOG_ERR,"[STATISTICS] Failed to get SID from identifier '%s': %s",$identifier,awitpt::db::dblayer::Error());
} }
return undef; return;
} }
# Set SID from identifier in DB # Set SID from identifier in DB
sub _setSIDFromIdentifier sub _setSIDFromIdentifier
{ {
my $identifier = shift; my $identifier = shift;
# We need the DB to be alive to do this...
if (!defined($globals->{'Database'})) {
return;
}
# Try add it to the DB # Try add it to the DB
my $identifierAddSTH = $statsPreparedStatements->{'identifier_add'}; if (my $res = DBDo(SQL_ADD_IDENTIFIER,$identifier)) {
if (my $res = $identifierAddSTH->execute($identifier)) { return $globals->{'IdentifierMap'}->{$identifier} = DBLastInsertID("","");
return $statsDBIdentifierMap->{$identifier} = $dbh->last_insert_id("","","","");
} else { } else {
$logger->log(LOG_ERR,"[STATISTICS] Failed to get SID from identifier '%s': %s",$identifier,$identifierAddSTH->errstr); $logger->log(LOG_ERR,"[STATISTICS] Failed to set SID from identifier '%s': %s",$identifier,awitpt::db::dblayer::Error());
} }
return undef; return;
} }
# Get aligned time on a Precision # Get aligned time on a Precision
sub _getAlignedTime sub _getAlignedTime
{ {
...@@ -1026,95 +1191,173 @@ sub _getAlignedTime ...@@ -1026,95 +1191,173 @@ sub _getAlignedTime
} }
# Internal function to get stats by SID # Internal function to get stats by SID
sub _getStatsBySID sub _getStatsBySID
{ {
my $sid = shift; my ($sid,$startTimestamp,$endTimestamp) = @_;
my $now = time(); my $now = time();
# Prepare query # Setup our timestamps if we need to
my $sth = $dbh->prepare(' if (!defined($startTimestamp)) {
SELECT $startTimestamp = $now - 3600;
`Timestamp`, `Direction`, `Rate`, `PPS`, `CIR`, `Limit` }
FROM if (!defined($endTimestamp)) {
stats $endTimestamp = $now;
WHERE }
`IdentifierID` = ?
AND `Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
ORDER BY
`Timestamp` DESC
LIMIT 100
');
# Grab last 60 mins of data
$sth->execute($sid,0,$now - 3600, $now);
my $statistics; # Work out the timestamp
while (my $item = $sth->fetchrow_hashref()) { my $timespan = $endTimestamp - $startTimestamp;
# Make direction a bit easier to use
my $direction; # Find the best key to use...
if ($item->{'direction'} eq STATISTICS_DIR_TX) { my $statsKey = 0;
$direction = 'tx'; foreach my $key (sort {$b <=> $a} keys %{STATS_CONFIG()}) {
} elsif ($item->{'direction'} eq STATISTICS_DIR_RX) { # Grab first key that will hve 50+ entries
$direction = 'rx'; if ($timespan / STATS_CONFIG()->{$key}->{'precision'} > 50) {
} else { $statsKey = $key;
$logger->log(LOG_ERR,"[STATISTICS] Unknown direction when getting stats '%s'",$direction); last;
next;
} }
}
my $statistics = { };
# We need the DB below this point
if (!defined($globals->{'Database'})) {
return $statistics;
}
# Loop with both directions # Grab last 60 mins of data
$statistics->{$item->{'timestamp'}}->{$direction} = { my $res = DBSelect(SQL_GET_STATS,$sid,$statsKey,$startTimestamp,$endTimestamp);
if (!defined($res)) {
$logger->log(LOG_ERR,"[STATISTICS] Failed to get stats for SID '%s': %s",$sid,awitpt::db::dblayer::Error());
return $statistics;
}
while (my $item = $res->fetchrow_hashref()) {
$statistics->{$item->{'timestamp'}}->{$item->{'direction'}} = {
'rate' => $item->{'rate'}, 'rate' => $item->{'rate'},
'pps' => $item->{'pps'}, 'pps' => $item->{'pps'},
'cir' => $item->{'cir'}, 'cir' => $item->{'cir'},
'limit' => $item->{'limit'}, 'limit' => $item->{'limit'},
} }
} }
DBFreeRes($res);
return $statistics; return $statistics;
} }
# Internal function to get basic stats by SID # Internal function to get basic stats by SID
sub _getStatsBasicBySID sub _getStatsBasicBySID
{ {
my $sid = shift; my ($sid,$startTimestamp,$endTimestamp) = @_;
my $now = time(); my $now = time();
# Setup our timestamps if we need to
if (!defined($startTimestamp)) {
$startTimestamp = $now - 3600;
}
if (!defined($endTimestamp)) {
$endTimestamp = $now;
}
# Work out the timestamp
my $timespan = $endTimestamp - $startTimestamp;
# Find the best key to use...
my $statsKey = 0;
foreach my $key (sort {$b <=> $a} keys %{STATS_CONFIG()}) {
# Grab first key that will hve 50+ entries
if ($timespan / STATS_CONFIG()->{$key}->{'precision'} > 50) {
$statsKey = $key;
last;
}
}
my $statistics = { };
# We need the DB below this point
if (!defined($globals->{'Database'})) {
return $statistics;
}
# Prepare query # Prepare query
my $sth = $dbh->prepare(' my $res = DBSelect(SQL_GET_STATS_BASIC,$sid,$statsKey,$startTimestamp,$endTimestamp);
SELECT
`Timestamp`, `Counter`
FROM
stats_basic
WHERE
`IdentifierID` = ?
AND `Key` = ?
AND `Timestamp` > ?
AND `Timestamp` < ?
ORDER BY
`Timestamp` DESC
LIMIT 100
');
# Grab last 60 mins of data
$sth->execute($sid,0,$now - 3600, $now);
my $statistics; while (my $item = $res->fetchrow_hashref()) {
while (my $item = $sth->fetchrow_hashref()) {
$statistics->{$item->{'timestamp'}} = { $statistics->{$item->{'timestamp'}} = {
'counter' => $item->{'counter'}, 'counter' => $item->{'counter'},
} }
} }
DBFreeRes($res);
return $statistics; return $statistics;
} }
# Function to transform stats before sending them
sub _fixStatDirection
{
my ($stat,$conversions) = @_;
my $res;
# Loop with directions, maybe we have more than one with this stat
while ((my $direction, my $oldStat) = each(%{$stat})) {
# Depending which direction, grab the key to use below
my $oldKey;
if ($direction == STATISTICS_DIR_TX) {
$oldKey = 'tx';
} elsif ($direction == STATISTICS_DIR_RX) {
$oldKey = 'rx';
}
# Loop and remove the direction, instead, adding it to the item
foreach my $item (keys %{$oldStat}) {
# If we have conversions defined...
my $newKey;
if (defined($conversions) && defined($conversions->{'Direction'})) {
$newKey = sprintf("%s.%s",$conversions->{'Direction'},$item);
} else {
$newKey = sprintf("%s.%s",$oldKey,$item);
}
$res->{$newKey} = $oldStat->{$item};
}
}
return $res;
}
# Function to transform stats before sending them
sub _fixCounterName
{
my ($stat,$conversions) = @_;
# Loop and set the identifier
my $newStat;
# If we have conversions defined...
my $newKey = 'counter';
if (defined($conversions) && defined($conversions->{'Name'})) {
$newKey = sprintf('%s',$conversions->{'Name'});
}
$newStat->{$newKey} = $stat->{'counter'};
return $newStat;
}
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper Linux tc traffic shaping # OpenTrafficShaper Linux tc traffic shaping
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -21,41 +21,52 @@ package opentrafficshaper::plugins::tc; ...@@ -21,41 +21,52 @@ package opentrafficshaper::plugins::tc;
use strict; use strict;
use warnings; use warnings;
use JSON;
use List::Util qw(min max);
use POE qw(
Wheel::Run Filter::Line
);
use POE qw( Wheel::Run Filter::Line ); use awitpt::util qw(
toHex
);
use opentrafficshaper::constants; use opentrafficshaper::constants;
use opentrafficshaper::logger; use opentrafficshaper::logger;
use opentrafficshaper::utils;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
getPool getPool
getPoolAttribute getPoolAttribute
setPoolAttribute setPoolAttribute
removePoolAttribute removePoolAttribute
getPoolTxInterface getPoolTxInterface
getPoolRxInterface getPoolRxInterface
setPoolShaperState setPoolShaperState
unsetPoolShaperState unsetPoolShaperState
getPoolShaperState getPoolShaperState
getEffectivePool getEffectivePool
getPoolMember getPoolMember
setPoolMemberAttribute setPoolMemberAttribute
getPoolMemberAttribute getPoolMemberAttribute
removePoolMemberAttribute removePoolMemberAttribute
getPoolMemberMatchPriority getPoolMemberMatchPriority
setPoolMemberShaperState setPoolMemberShaperState
unsetPoolMemberShaperState unsetPoolMemberShaperState
getPoolMemberShaperState getPoolMemberShaperState
getTrafficClassPriority getTrafficClassPriority
getInterfaces getAllTrafficClasses
getInterfaceRate
getInterfaceTrafficClasses getInterface
getInterfaceDefaultPool getInterfaceGroup
getInterfaceGroups
getInterfaces
getInterfaceDefaultPool
getEffectiveInterfaceTrafficClass2
isInterfaceTrafficClassValid
setInterfaceTrafficClassShaperState
unsetInterfaceTrafficClassShaperState
); );
...@@ -69,7 +80,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -69,7 +80,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
); );
use constant { use constant {
VERSION => '0.1.2', VERSION => '1.0.1',
# 5% of a link can be used for very high priority traffic # 5% of a link can be used for very high priority traffic
PROTO_RATE_LIMIT => 5, PROTO_RATE_LIMIT => 5,
...@@ -95,8 +106,9 @@ our $pluginInfo = { ...@@ -95,8 +106,9 @@ our $pluginInfo = {
}; };
# Copy of system globals # Our globals
my $globals; my $globals;
# Copy of system logger
my $logger; my $logger;
# Our configuration # Our configuration
...@@ -105,33 +117,42 @@ my $config = { ...@@ -105,33 +117,42 @@ my $config = {
'iphdr_offset' => 0, 'iphdr_offset' => 0,
}; };
# Queue of tasks to run #
my @taskQueue = ( ); # TASK QUEUE
# TC classes & filters #
my $tcClasses = { }; # $globals->{'TaskQueue'}
my $tcFilterMappings;
my $tcFilters = { };
#
# TC CLASSES & FILTERS
#
# $globals->{'TcClasses'}
# $globals->{'TcFilterMappings'}
# $globals->{'TcFilters'}
# Initialize plugin # Initialize plugin
sub plugin_init sub plugin_init
{ {
$globals = shift; my $system = shift;
# Setup our environment # Setup our environment
$logger = $globals->{'logger'}; $logger = $system->{'logger'};
$logger->log(LOG_NOTICE,"[TC] OpenTrafficShaper tc Integration v%s - Copyright (c) 2007-2014, AllWorldIT",VERSION); $logger->log(LOG_NOTICE,"[TC] OpenTrafficShaper tc Integration v%s - Copyright (c) 2007-2023, AllWorldIT",VERSION);
# Initialize
$globals->{'TaskQueue'} = [ ];
$globals->{'TcClasses'} = { };
$globals->{'TcFilterMappings'} = { };
$globals->{'TcFilters'} = { };
# Grab some of our config we need # Grab some of our config we need
if (defined(my $proto = $globals->{'file.config'}->{'plugin.tc'}->{'protocol'})) { if (defined(my $proto = $system->{'file.config'}->{'plugin.tc'}->{'protocol'})) {
$logger->log(LOG_INFO,"[TC] Set protocol to '%s'",$proto); $logger->log(LOG_INFO,"[TC] Set protocol to '%s'",$proto);
$config->{'ip_protocol'} = $proto; $config->{'ip_protocol'} = $proto;
} }
if (defined(my $offset = $globals->{'file.config'}->{'plugin.tc'}->{'iphdr_offset'})) { if (defined(my $offset = $system->{'file.config'}->{'plugin.tc'}->{'iphdr_offset'})) {
$logger->log(LOG_INFO,"[TC] Set IP header offset to '%s'",$offset); $logger->log(LOG_INFO,"[TC] Set IP header offset to '%s'",$offset);
$config->{'iphdr_offset'} = $offset; $config->{'iphdr_offset'} = $offset;
} }
...@@ -139,11 +160,261 @@ sub plugin_init ...@@ -139,11 +160,261 @@ sub plugin_init
# We going to queue the initialization in plugin initialization so nothing at all can come before us # We going to queue the initialization in plugin initialization so nothing at all can come before us
my $changeSet = TC::ChangeSet->new(); my $changeSet = TC::ChangeSet->new();
# Loop with protocols
for my $ipv ("", "6") {
#
# Traffic Shaping
#
# First the Cleanup
#
foreach my $interfaceID (getInterfaces()) {
my $interface = getInterface($interfaceID);
# Flush inteface chain
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-F',"ots-tcfor-$interfaceID",
]);
# Delete jump
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-D','ots-tcfor',
'-o',$interface->{'Device'},
'-j',"ots-tcfor-$interfaceID",
]);
# Delete interface chain
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-X',"ots-tcfor-$interfaceID",
]);
}
# Flush ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-F','ots-tcfor',
]);
# Delete jump to ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-D','FORWARD',
'-j','ots-tcfor',
]);
# Delete ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-X','ots-tcfor',
]);
# Then the setup
#
# Create ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-N','ots-tcfor',
]);
# Add jump to ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-I','FORWARD',
'-j','ots-tcfor',
]);
# Add interface chains
foreach my $interfaceID (getInterfaces()) {
my $interface = getInterface($interfaceID);
# Create ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-N',"ots-tcfor-$interfaceID",
]);
# Add jump to ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','mangle',
'-A','ots-tcfor',
'-o',$interface->{'Device'},
'-j',"ots-tcfor-$interfaceID",
]);
}
#
# NAT
#
# First the Cleanup
#
foreach my $interfaceGroupName (getInterfaceGroups()) {
my $interfaceGroup = getInterfaceGroup($interfaceGroupName);
my $txInterfaceID = $interfaceGroup->{'TxInterface'};
my $rxInterfaceID = $interfaceGroup->{'RxInterface'};
my $txInterface = getInterface($txInterfaceID);
my $rxInterface = getInterface($rxInterfaceID);
# Flush inteface chains
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-F',"ots-snat-$txInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-F',"ots-dnat-$rxInterfaceID",
]);
# Delete interface chains
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-X',"ots-snat-$txInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-X',"ots-dnat-$rxInterfaceID",
]);
}
# Flush ots-snat and ots-dnat
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-F','ots-snat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-F','ots-dnat',
]);
# Delete jump to ots-snat and ots-dnat
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-D','POSTROUTING',
'-j','ots-snat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-D','OUTPUT',
'-j','ots-dnat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-D','PREROUTING',
'-j','ots-dnat',
]);
# Delete ots-tcfor
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-X','ots-snat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-X','ots-dnat',
]);
# Then the setup
#
# Create ots-snat and ots-dnat
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-N','ots-snat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-N','ots-dnat',
]);
# Add jump to ots-snat and ots-dnat
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-I','POSTROUTING',
'-j','ots-snat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-I','OUTPUT',
'-j','ots-dnat',
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A','PREROUTING',
'-j','ots-dnat',
]);
foreach my $interfaceGroupName (getInterfaceGroups()) {
my $interfaceGroup = getInterfaceGroup($interfaceGroupName);
my $txInterfaceID = $interfaceGroup->{'TxInterface'};
my $rxInterfaceID = $interfaceGroup->{'RxInterface'};
my $txInterface = getInterface($txInterfaceID);
my $rxInterface = getInterface($rxInterfaceID);
# Flush inteface chain
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-N',"ots-snat-$txInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-N',"ots-dnat-$rxInterfaceID",
]);
# Add interface chains
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A','ots-snat',
'-o',$rxInterface->{'Device'},
'-j',"ots-snat-$txInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A','ots-snat',
'-o',$txInterface->{'Device'},
'-j',"ots-snat-$txInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A','ots-dnat',
'-i',$rxInterface->{'Device'},
'-j',"ots-dnat-$rxInterfaceID",
]);
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A','ots-dnat',
'-i',$txInterface->{'Device'},
'-j',"ots-dnat-$rxInterfaceID",
]);
}
}
# Loop with the configured interfaces and initialize them # Loop with the configured interfaces and initialize them
foreach my $interface (@{getInterfaces()}) { foreach my $interfaceID (getInterfaces()) {
my $interface = getInterface($interfaceID);
# Initialize interface # Initialize interface
$logger->log(LOG_INFO,"[TC] Queuing tasks to initialize '%s'",$interface); $logger->log(LOG_INFO,"[TC] Queuing tasks to initialize '%s'",$interface->{'Device'});
_tc_iface_init($changeSet,$interface); _tc_iface_init($changeSet,$interfaceID);
} }
_task_add_to_queue($changeSet); _task_add_to_queue($changeSet);
...@@ -154,6 +425,8 @@ sub plugin_init ...@@ -154,6 +425,8 @@ sub plugin_init
_start => \&_session_start, _start => \&_session_start,
_stop => \&_session_stop, _stop => \&_session_stop,
class_change => \&_session_class_change,
pool_add => \&_session_pool_add, pool_add => \&_session_pool_add,
pool_remove => \&_session_pool_remove, pool_remove => \&_session_pool_remove,
pool_change => \&_session_pool_change, pool_change => \&_session_pool_change,
...@@ -189,6 +462,7 @@ sub plugin_init ...@@ -189,6 +462,7 @@ sub plugin_init
} }
# Start the plugin # Start the plugin
sub plugin_start sub plugin_start
{ {
...@@ -196,6 +470,7 @@ sub plugin_start ...@@ -196,6 +470,7 @@ sub plugin_start
} }
# Initialize this plugins main POE session # Initialize this plugins main POE session
sub _session_start sub _session_start
{ {
...@@ -209,6 +484,7 @@ sub _session_start ...@@ -209,6 +484,7 @@ sub _session_start
} }
# Initialize this plugins main POE session # Initialize this plugins main POE session
sub _session_stop sub _session_stop
{ {
...@@ -220,9 +496,6 @@ sub _session_stop ...@@ -220,9 +496,6 @@ sub _session_stop
# Blow away data # Blow away data
$globals = undef; $globals = undef;
@taskQueue = ();
$tcFilterMappings = undef;
# XXX: Destroy the rest too like config
$logger->log(LOG_DEBUG,"[TC] Shutdown"); $logger->log(LOG_DEBUG,"[TC] Shutdown");
...@@ -230,6 +503,62 @@ sub _session_stop ...@@ -230,6 +503,62 @@ sub _session_stop
} }
# Event handler for changing a class
sub _session_class_change
{
my ($kernel, $interfaceTrafficClassID) = @_[KERNEL, ARG0, ARG1];
# Grab our effective class
my $effectiveInterfaceTrafficClass = getEffectiveInterfaceTrafficClass2($interfaceTrafficClassID);
# Grab interface ID
my $interfaceID = $effectiveInterfaceTrafficClass->{'InterfaceID'};
# Grab interface from config manager
my $interface = getInterface($interfaceID);
# Grab traffic class ID
my $trafficClassID = $effectiveInterfaceTrafficClass->{'TrafficClassID'};
$logger->log(LOG_INFO,"[TC] Processing interface class changes for '%s' traffic class ID '%s'",
$interface->{'Device'},
$trafficClassID
);
# Grab tc interface
my $tcInterface = $globals->{'Interfaces'}->{$interfaceID};
# Grab interface traffic class
my $interfaceTrafficClass = $tcInterface->{'TrafficClasses'}->{$trafficClassID};
# Grab the traffic class
my $majorTcClass = $tcInterface->{'TcClass'};
my $minorTcClass = $interfaceTrafficClass->{"TcClass"};
# Generate changeset
my $changeSet = TC::ChangeSet->new();
# If we're a normal class we are treated differently than if we're a main/root class below (interface main speed)
if ($minorTcClass > 1) {
_tc_class_change($changeSet,$interfaceID,$majorTcClass,"",$minorTcClass,
$effectiveInterfaceTrafficClass->{'CIR'},
$effectiveInterfaceTrafficClass->{'Limit'}
);
# XXX: This will be the actual interface, we set limit and burst to the same
} else {
_tc_class_change($changeSet,$interfaceID,TC_ROOT_CLASS,"",$minorTcClass,$effectiveInterfaceTrafficClass->{'Limit'});
}
# Post changeset
$kernel->post("_tc" => "queue" => $changeSet);
# Mark as live
unsetInterfaceTrafficClassShaperState($interfaceTrafficClassID,SHAPER_NOTLIVE|SHAPER_PENDING);
setInterfaceTrafficClassShaperState($interfaceTrafficClassID,SHAPER_LIVE);
}
# Event handler for adding a pool # Event handler for adding a pool
sub _session_pool_add sub _session_pool_add
{ {
...@@ -239,14 +568,14 @@ sub _session_pool_add ...@@ -239,14 +568,14 @@ sub _session_pool_add
# Grab pool # Grab pool
my $pool; my $pool;
if (!defined($pool = getPool($pid))) { if (!defined($pool = getPool($pid))) {
$logger->log(LOG_ERR,"[TC] Shaper 'remove' event with non existing pool '%s'",$pid); $logger->log(LOG_ERR,"[TC] Shaper 'add' event with non existing pool '%s'",$pid);
return; return;
} }
$logger->log(LOG_INFO,"[TC] Add pool '%s' to interface group '%s' [%s]", $logger->log(LOG_INFO,"[TC] Add pool '%s' [%s] to interface group '%s'",
$pool->{'Name'}, $pool->{'Name'},
$pool->{'ID'},
$pool->{'InterfaceGroupID'}, $pool->{'InterfaceGroupID'},
$pool->{'ID'}
); );
# Grab our effective pool # Grab our effective pool
...@@ -255,39 +584,40 @@ sub _session_pool_add ...@@ -255,39 +584,40 @@ sub _session_pool_add
my $changeSet = TC::ChangeSet->new(); my $changeSet = TC::ChangeSet->new();
# Grab some things we need from the main pool # Grab some things we need from the main pool
my $txInterface = getPoolTxInterface($pool->{'ID'}); my $txInterfaceID = getPoolTxInterface($pool->{'ID'});
my $rxInterface = getPoolRxInterface($pool->{'ID'}); my $rxInterfaceID = getPoolRxInterface($pool->{'ID'});
# Grab effective config # Grab effective config
my $classID = $effectivePool->{'ClassID'}; my $trafficClassID = $effectivePool->{'TrafficClassID'};
my $trafficLimitTx = $effectivePool->{'TrafficLimitTx'}; my $txCIR = $effectivePool->{'TxCIR'};
my $trafficLimitTxBurst = $effectivePool->{'TrafficLimitTxBurst'}; my $txLimit = $effectivePool->{'TxLimit'};
my $trafficLimitRx = $effectivePool->{'TrafficLimitRx'}; my $rxCIR = $effectivePool->{'RxCIR'};
my $trafficLimitRxBurst = $effectivePool->{'TrafficLimitRxBurst'}; my $rxLimit = $effectivePool->{'RxLimit'};
my $trafficPriority = getTrafficClassPriority($effectivePool->{'ClassID'}); my $trafficPriority = getTrafficClassPriority($effectivePool->{'TrafficClassID'});
# Get the Tx traffic classes TC class # Get the Tx traffic classes TC class
my $tcClass_TxTrafficClass = _getTcClassFromTrafficClassID($txInterface,$classID); my $tcClass_TxTrafficClass = _getTcClassFromTrafficClassID($txInterfaceID,$trafficClassID);
# Generate our pools Tx TC class # Generate our pools Tx TC class
my $tcClass_TxPool = _reserveTcClassByPoolID($txInterface,$pool->{'ID'}); my $tcClass_TxPool = _reserveMinorTcClassByPoolID($txInterfaceID,$pool->{'ID'});
# Add the main Tx TC class for this pool # Add the main Tx TC class for this pool
_tc_class_add($changeSet,$txInterface,TC_ROOT_CLASS,$tcClass_TxTrafficClass,$tcClass_TxPool,$trafficLimitTx, _tc_class_add($changeSet,$txInterfaceID,TC_ROOT_CLASS,$tcClass_TxTrafficClass,$tcClass_TxPool,$txCIR,
$trafficLimitTxBurst,$trafficPriority $txLimit,$trafficPriority
); );
# Add Tx TC optimizations # Add Tx TC optimizations
_tc_class_optimize($changeSet,$txInterface,$tcClass_TxPool,$trafficLimitTx); _tc_class_optimize($changeSet,$txInterfaceID,$tcClass_TxPool,$txCIR);
# Set Tx TC class # Set Tx TC class
setPoolAttribute($pool->{'ID'},'tc.txclass',$tcClass_TxPool); setPoolAttribute($pool->{'ID'},'tc.txclass',$tcClass_TxPool);
# Get the Rx traffic classes TC class # Get the Rx traffic classes TC class
my $tcClass_RxTrafficClass = _getTcClassFromTrafficClassID($rxInterface,$classID); my $tcClass_RxTrafficClass = _getTcClassFromTrafficClassID($rxInterfaceID,$trafficClassID);
# Generate our pools Rx TC class # Generate our pools Rx TC class
my $tcClass_RxPool = _reserveTcClassByPoolID($rxInterface,$pool->{'ID'}); my $tcClass_RxPool = _reserveMinorTcClassByPoolID($rxInterfaceID,$pool->{'ID'});
# Add the main Rx TC class for this pool # Add the main Rx TC class for this pool
_tc_class_add($changeSet,$rxInterface,TC_ROOT_CLASS,$tcClass_RxTrafficClass,$tcClass_RxPool,$trafficLimitRx, _tc_class_add($changeSet,$rxInterfaceID,TC_ROOT_CLASS,$tcClass_RxTrafficClass,$tcClass_RxPool,$rxCIR,
$trafficLimitRxBurst,$trafficPriority $rxLimit,$trafficPriority
); );
# Add Rx TC optimizations # Add Rx TC optimizations
_tc_class_optimize($changeSet,$rxInterface,$tcClass_RxPool,$trafficLimitRx); _tc_class_optimize($changeSet,$rxInterfaceID,$tcClass_RxPool,$rxCIR);
# Set Rx TC # Set Rx TC
setPoolAttribute($pool->{'ID'},'tc.rxclass',$tcClass_RxPool); setPoolAttribute($pool->{'ID'},'tc.rxclass',$tcClass_RxPool);
...@@ -295,11 +625,11 @@ sub _session_pool_add ...@@ -295,11 +625,11 @@ sub _session_pool_add
$kernel->post("_tc" => "queue" => $changeSet); $kernel->post("_tc" => "queue" => $changeSet);
# Set current live values # Set current live values
setPoolAttribute($pool->{'ID'},'shaper.live.ClassID',$classID); setPoolAttribute($pool->{'ID'},'shaper.live.ClassID',$trafficClassID);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTx',$trafficLimitTx); setPoolAttribute($pool->{'ID'},'shaper.live.TxCIR',$txCIR);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTxBurst',$trafficLimitTxBurst); setPoolAttribute($pool->{'ID'},'shaper.live.TxLimit',$txLimit);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRx',$trafficLimitRx); setPoolAttribute($pool->{'ID'},'shaper.live.RxCIR',$rxCIR);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRxBurst',$trafficLimitRxBurst); setPoolAttribute($pool->{'ID'},'shaper.live.RxLimit',$rxLimit);
# Mark as live # Mark as live
unsetPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE|SHAPER_PENDING); unsetPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE|SHAPER_PENDING);
...@@ -307,6 +637,7 @@ sub _session_pool_add ...@@ -307,6 +637,7 @@ sub _session_pool_add
} }
# Event handler for removing a pool # Event handler for removing a pool
sub _session_pool_remove sub _session_pool_remove
{ {
...@@ -337,38 +668,41 @@ sub _session_pool_remove ...@@ -337,38 +668,41 @@ sub _session_pool_remove
); );
# Grab our interfaces # Grab our interfaces
my $txInterface = getPoolTxInterface($pool->{'ID'}); my $txInterfaceID = getPoolTxInterface($pool->{'ID'});
my $rxInterface = getPoolRxInterface($pool->{'ID'}); my $rxInterfaceID = getPoolRxInterface($pool->{'ID'});
# Grab the traffic class from the pool # Grab the traffic class from the pool
my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass'); my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass');
my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass'); my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass');
# Grab current class ID # Grab current class ID
my $classID = getPoolAttribute($pool->{'ID'},'shaper.live.ClassID'); my $trafficClassID = getPoolAttribute($pool->{'ID'},'shaper.live.ClassID');
# Grab our minor classes # Grab our minor classes
my $txTrafficClassTcClass = _getTcClassFromTrafficClassID($txInterface,$classID); my $txTrafficClassTcClass = _getTcClassFromTrafficClassID($txInterfaceID,$trafficClassID);
my $rxTrafficClassTcClass = _getTcClassFromTrafficClassID($rxInterface,$classID); my $rxTrafficClassTcClass = _getTcClassFromTrafficClassID($rxInterfaceID,$trafficClassID);
my $txInterface = getInterface($txInterfaceID);
my $rxInterface = getInterface($rxInterfaceID);
# Clear up the class # Clear up the class
$changeSet->add([ $changeSet->add([
'/sbin/tc','class','del', '/sbin/tc','class','del',
'dev',$txInterface, 'dev',$txInterface->{'Device'},
'parent',"1:$txTrafficClassTcClass", 'parent',"1:$txTrafficClassTcClass",
'classid',"1:$txPoolTcClass", 'classid',"1:$txPoolTcClass",
]); ]);
$changeSet->add([ $changeSet->add([
'/sbin/tc','class','del', '/sbin/tc','class','del',
'dev',$rxInterface, 'dev',$rxInterface->{'Device'},
'parent',"1:$rxTrafficClassTcClass", 'parent',"1:$rxTrafficClassTcClass",
'classid',"1:$rxPoolTcClass", 'classid',"1:$rxPoolTcClass",
]); ]);
# And recycle the classs # And recycle the classs
_disposePoolTcClass($txInterface,$txPoolTcClass); _disposePoolTcClass($txInterface->{'Device'},$txPoolTcClass);
_disposePoolTcClass($rxInterface,$rxPoolTcClass); _disposePoolTcClass($rxInterface->{'Device'},$rxPoolTcClass);
_disposePrioTcClass($txInterface,$txPoolTcClass); _disposePrioTcClass($txInterface->{'Device'},$txPoolTcClass);
_disposePrioTcClass($rxInterface,$rxPoolTcClass); _disposePrioTcClass($rxInterface->{'Device'},$rxPoolTcClass);
# Post changeset # Post changeset
$kernel->post("_tc" => "queue" => $changeSet); $kernel->post("_tc" => "queue" => $changeSet);
...@@ -378,10 +712,10 @@ sub _session_pool_remove ...@@ -378,10 +712,10 @@ sub _session_pool_remove
removePoolAttribute($pool->{'ID'},'tc.rxclass'); removePoolAttribute($pool->{'ID'},'tc.rxclass');
removePoolAttribute($pool->{'ID'},'shaper.live.ClassID'); removePoolAttribute($pool->{'ID'},'shaper.live.ClassID');
removePoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTx'); removePoolAttribute($pool->{'ID'},'shaper.live.TxCIR');
removePoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTxBurst'); removePoolAttribute($pool->{'ID'},'shaper.live.TxLimit');
removePoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRx'); removePoolAttribute($pool->{'ID'},'shaper.live.RxCIR');
removePoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRxBurst'); removePoolAttribute($pool->{'ID'},'shaper.live.RxLimit');
# Mark as not live # Mark as not live
unsetPoolShaperState($pool->{'ID'},SHAPER_LIVE|SHAPER_PENDING); unsetPoolShaperState($pool->{'ID'},SHAPER_LIVE|SHAPER_PENDING);
...@@ -389,10 +723,11 @@ sub _session_pool_remove ...@@ -389,10 +723,11 @@ sub _session_pool_remove
} }
## Event handler for changing a pool ## Event handler for changing a pool
sub _session_pool_change sub _session_pool_change
{ {
my ($kernel, $pid) = @_[KERNEL, ARG0, ARG1]; my ($kernel, $pid) = @_[KERNEL, ARG0];
# Grab pool # Grab pool
...@@ -404,40 +739,40 @@ sub _session_pool_change ...@@ -404,40 +739,40 @@ sub _session_pool_change
my $effectivePool = getEffectivePool($pool->{'ID'}); my $effectivePool = getEffectivePool($pool->{'ID'});
# Grab our interfaces # Grab our interfaces
my $txInterface = getPoolTxInterface($pool->{'ID'}); my $txInterfaceID = getPoolTxInterface($pool->{'ID'});
my $rxInterface = getPoolRxInterface($pool->{'ID'}); my $rxInterfaceID = getPoolRxInterface($pool->{'ID'});
# Grab the traffic class from the pool # Grab the traffic class from the pool
my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass'); my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass');
my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass'); my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass');
# Grab effective config # Grab effective config
my $classID = $effectivePool->{'ClassID'}; my $trafficClassID = $effectivePool->{'TrafficClassID'};
my $trafficLimitTx = $effectivePool->{'TrafficLimitTx'}; my $txCIR = $effectivePool->{'TxCIR'};
my $trafficLimitTxBurst = $effectivePool->{'TrafficLimitTxBurst'}; my $txLimit = $effectivePool->{'TxLimit'};
my $trafficLimitRx = $effectivePool->{'TrafficLimitRx'}; my $rxCIR = $effectivePool->{'RxCIR'};
my $trafficLimitRxBurst = $effectivePool->{'TrafficLimitRxBurst'}; my $rxLimit = $effectivePool->{'RxLimit'};
my $trafficPriority = getTrafficClassPriority($classID); my $trafficPriority = getTrafficClassPriority($trafficClassID);
# Grab our minor classes # Grab our minor classes
my $txTrafficClassTcClass = _getTcClassFromTrafficClassID($txInterface,$classID); my $txTrafficClassTcClass = _getTcClassFromTrafficClassID($txInterfaceID,$trafficClassID);
my $rxTrafficClassTcClass = _getTcClassFromTrafficClassID($rxInterface,$classID); my $rxTrafficClassTcClass = _getTcClassFromTrafficClassID($rxInterfaceID,$trafficClassID);
# Generate changeset # Generate changeset
my $changeSet = TC::ChangeSet->new(); my $changeSet = TC::ChangeSet->new();
_tc_class_change($changeSet,$txInterface,TC_ROOT_CLASS,$txTrafficClassTcClass,$txPoolTcClass,$trafficLimitTx, _tc_class_change($changeSet,$txInterfaceID,TC_ROOT_CLASS,$txTrafficClassTcClass,$txPoolTcClass,$txCIR,
$trafficLimitTxBurst,$trafficPriority); $txLimit,$trafficPriority);
_tc_class_change($changeSet,$rxInterface,TC_ROOT_CLASS,$rxTrafficClassTcClass,$rxPoolTcClass,$trafficLimitRx, _tc_class_change($changeSet,$rxInterfaceID,TC_ROOT_CLASS,$rxTrafficClassTcClass,$rxPoolTcClass,$rxCIR,
$trafficLimitRxBurst,$trafficPriority); $rxLimit,$trafficPriority);
# Post changeset # Post changeset
$kernel->post("_tc" => "queue" => $changeSet); $kernel->post("_tc" => "queue" => $changeSet);
setPoolAttribute($pool->{'ID'},'shaper.live.ClassID',$classID); setPoolAttribute($pool->{'ID'},'shaper.live.ClassID',$trafficClassID);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTx',$trafficLimitTx); setPoolAttribute($pool->{'ID'},'shaper.live.TxCIR',$txCIR);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitTxBurst',$trafficLimitTxBurst); setPoolAttribute($pool->{'ID'},'shaper.live.TxLimit',$txLimit);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRx',$trafficLimitRx); setPoolAttribute($pool->{'ID'},'shaper.live.RxCIR',$rxCIR);
setPoolAttribute($pool->{'ID'},'shaper.live.TrafficLimitRxBurst',$trafficLimitRxBurst); setPoolAttribute($pool->{'ID'},'shaper.live.RxLimit',$rxLimit);
# Mark as live # Mark as live
unsetPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE|SHAPER_PENDING); unsetPoolShaperState($pool->{'ID'},SHAPER_NOTLIVE|SHAPER_PENDING);
...@@ -445,6 +780,7 @@ sub _session_pool_change ...@@ -445,6 +780,7 @@ sub _session_pool_change
} }
# Event handler for adding a pool member # Event handler for adding a pool member
sub _session_poolmember_add sub _session_poolmember_add
{ {
...@@ -458,188 +794,102 @@ sub _session_poolmember_add ...@@ -458,188 +794,102 @@ sub _session_poolmember_add
return; return;
} }
$logger->log(LOG_INFO,"[TC] Add pool member '%s' to pool '%s' [%s]", # Grab the pool members associated pool
$poolMember->{'IPAddress'},
$poolMember->{'PoolID'},
$poolMember->{'ID'}
);
my $changeSet = TC::ChangeSet->new();
# Filter levels for the IP components
my @components = split(/\./,$poolMember->{'IPAddress'});
my $ip1 = $components[0];
my $ip2 = $components[1];
my $ip3 = $components[2];
my $ip4 = $components[3];
my $pool; my $pool;
if (!defined($pool = getPool($poolMember->{'PoolID'}))) { if (!defined($pool = getPool($poolMember->{'PoolID'}))) {
$logger->log(LOG_ERR,"[TC] Shaper 'poolmember_add' event with invalid PoolID"); $logger->log(LOG_ERR,"[TC] Shaper 'poolmember_add' event with invalid PoolID");
return; return;
} }
# Grab some variables we going to need below $logger->log(LOG_INFO,"[TC] Add pool member '%s' [%s] with IP '%s', NAT '%s' (inbound: %s) to pool '%s' [%s]",
my $txInterface = getPoolTxInterface($pool->{'ID'}); $poolMember->{'Username'},
my $rxInterface = getPoolRxInterface($pool->{'ID'}); $poolMember->{'ID'},
my $trafficPriority = getTrafficClassPriority($pool->{'ClassID'}); $poolMember->{'IPAddress'},
my $matchPriority = getPoolMemberMatchPriority($poolMember->{'ID'}); $poolMember->{'IPNATAddress'} // "",
$poolMember->{'IPNATInbound'} // "",
# Check if we have a entry for the /8, if not we must create our 2nd level hash table and link it $pool->{'Name'},
if (!defined($tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1})) { $pool->{'ID'}
# Grab filter ID's for 2nd level );
my $filterID = _reserveTcFilter($txInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{'id'} = $filterID;
$logger->log(LOG_DEBUG,"[TC] Linking 2nd level TX hash table to '%s' to '%s.0.0.0/8', priority '%s'",
$filterID,
$ip1,
$matchPriority
);
_tc_filter_add_dstlink($changeSet,$txInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},800,"",
"$ip1.0.0.0/8","00ff0000");
}
if (!defined($tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1})) {
# Grab filter ID's for 2nd level
my $filterID = _reserveTcFilter($rxInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{'id'} = $filterID;
$logger->log(LOG_DEBUG,"[TC] Linking 2nd level RX hash table to '%s' to '%s.0.0.0/8', priority '%s'",
$filterID,
$ip1,
$matchPriority
);
_tc_filter_add_srclink($changeSet,$rxInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},800,"",
"$ip1.0.0.0/8","00ff0000");
}
# Check if we have our /16 hash entry, if not we must create the 3rd level hash table
if (!defined($tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2})) {
# Grab filter ID's for 3rd level
my $filterID = _reserveTcFilter($txInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2}->{'id'} = $filterID;
# Grab some hash table ID's we need
my $ip1HtHex = $tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{'id'};
# And hex our IP component
my $ip2Hex = toHex($ip2);
$logger->log(LOG_DEBUG,"[TC] Linking 3rd level TX hash table to '%s' to '%s.%s.0.0/16', priority '%s'",
$filterID,
$ip1,
$ip2,
$matchPriority
);
_tc_filter_add_dstlink($changeSet,$txInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},$ip1HtHex,
$ip2Hex,"$ip1.$ip2.0.0/16","0000ff00");
}
if (!defined($tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2})) {
# Grab filter ID's for 3rd level
my $filterID = _reserveTcFilter($rxInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2}->{'id'} = $filterID;
# Grab some hash table ID's we need
my $ip1HtHex = $tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{'id'};
# And hex our IP component
my $ip2Hex = toHex($ip2);
$logger->log(LOG_DEBUG,"[TC] Linking 3rd level RX hash table to '%s' to '%s.%s.0.0/16', priority '%s'",
$filterID,
$ip1,
$ip2,
$matchPriority
);
_tc_filter_add_srclink($changeSet,$rxInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},$ip1HtHex,
$ip2Hex,"$ip1.$ip2.0.0/16","0000ff00");
}
# Check if we have our /24 hash entry, if not we must create the 4th level hash table my $changeSet = TC::ChangeSet->new();
if (!defined($tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3})) {
# Grab filter ID's for 4th level
my $filterID = _reserveTcFilter($txInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3}->{'id'} = $filterID;
# Grab some hash table ID's we need
my $ip2HtHex = $tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2}->{'id'};
# And hex our IP component
my $ip3Hex = toHex($ip3);
$logger->log(LOG_DEBUG,"[TC] Linking 4th level TX hash table to '%s' to '%s.%s.%s.0/24', priority '%s'",
$filterID,
$ip1,
$ip2,
$ip3,
$matchPriority
);
_tc_filter_add_dstlink($changeSet,$txInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},$ip2HtHex,
$ip3Hex,"$ip1.$ip2.$ip3.0/24","000000ff");
}
if (!defined($tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3})) {
# Grab filter ID's for 4th level
my $filterID = _reserveTcFilter($rxInterface,$matchPriority,$pool->{'ID'});
# Track our mapping
$tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3}->{'id'} = $filterID;
# Grab some hash table ID's we need
my $ip2HtHex = $tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2}->{'id'};
# And hex our IP component
my $ip3Hex = toHex($ip3);
$logger->log(LOG_DEBUG,"[TC] Linking 4th level RX hash table to '%s' to '%s.%s.%s.0/24', priority '%s'",
$filterID,
$ip1,
$ip2,
$ip3,
$matchPriority
);
_tc_filter_add_srclink($changeSet,$rxInterface,TC_ROOT_CLASS,$matchPriority,$filterID,$config->{'ip_protocol'},$ip2HtHex,
$ip3Hex,"$ip1.$ip2.$ip3.0/24","000000ff");
}
# my $txInterfaceID = getPoolTxInterface($pool->{'ID'});
# For sake of simplicity and so things loook all nice and similar, we going to do these 2 blocks in { } my $rxInterfaceID = getPoolRxInterface($pool->{'ID'});
#
# Only if we have TX limits setup process them
{
# Get the TX class
my $tcClass_trafficClass = getPoolAttribute($pool->{'ID'},'tc.txclass');
# Grab some hash table ID's we need
my $ip3HtHex = $tcFilterMappings->{$txInterface}->{'dst'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3}->{'id'};
# And hex our IP component
my $ip4Hex = toHex($ip4);
$logger->log(LOG_DEBUG,"[TC] Linking pool member IP '%s' to class '%s' at hash endpoint '%s:%s'",
$poolMember->{'IPAddress'},
$tcClass_trafficClass,
$ip3HtHex,
$ip4Hex
);
# Link filter to traffic flow (class) my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass');
_tc_filter_add_flowlink($changeSet,$txInterface,TC_ROOT_CLASS,$trafficPriority,$config->{'ip_protocol'},$ip3HtHex,$ip4Hex, my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass');
"dst",16,$poolMember->{'IPAddress'},$tcClass_trafficClass);
# Save pool member filter ID # Check what IP version we're dealing with
setPoolMemberAttribute($poolMember->{'ID'},'tc.txfilter',"${ip3HtHex}:${ip4Hex}:1"); my $ipv = "";
if ($poolMember->{'IPAddress'} =~ /:/) {
$ipv = "6";
} }
# Only if we have RX limits setup process them # Add traffic classification
{ $changeSet->add([
# Generate our limit TC class "/sbin/ip${ipv}tables",
my $tcClass_trafficClass = getPoolAttribute($pool->{'ID'},'tc.rxclass'); '-t','mangle',
# Grab some hash table ID's we need '-A',"ots-tcfor-$rxInterfaceID",
my $ip3HtHex = $tcFilterMappings->{$rxInterface}->{'src'}->{$matchPriority}->{$ip1}->{$ip2}->{$ip3}->{'id'}; '-s',$poolMember->{'IPAddress'},
# And hex our IP component '-m','comment',
my $ip4Hex = toHex($ip4); '--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
$logger->log(LOG_DEBUG,"[TC] Linking RX IP '%s' to class '%s' at hash endpoint '%s:%s'", '-j','CLASSIFY',
$poolMember->{'IPAddress'}, '--set-class',"1:$rxPoolTcClass",
$tcClass_trafficClass, ]);
$ip3HtHex, $changeSet->add([
$ip4Hex "/sbin/ip${ipv}tables",
); '-t','mangle',
'-A',"ots-tcfor-$txInterfaceID",
# Link filter to traffic flow (class) '-d',$poolMember->{'IPAddress'},
_tc_filter_add_flowlink($changeSet,$rxInterface,TC_ROOT_CLASS,$trafficPriority,$config->{'ip_protocol'},$ip3HtHex,$ip4Hex, '-m','comment',
"src",12,$poolMember->{'IPAddress'},$tcClass_trafficClass); '--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'-j','CLASSIFY',
# Save pool member filter ID '--set-class',"1:$txPoolTcClass",
setPoolMemberAttribute($poolMember->{'ID'},'tc.rxfilter',"${ip3HtHex}:${ip4Hex}:1"); ]);
# Add NAT if we have any
if (defined($poolMember->{'IPNATAddress'}) && defined($poolMember->{'IPNATAddress'}) ne "") {
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A',"ots-snat-$txInterfaceID",
'-s',$poolMember->{'IPAddress'},
'-m','comment',
'--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'-j','SNAT',
'--to-source', $poolMember->{'IPNATAddress'},
]);
# If we're dealing with IPv4, clear the connection tracking
if ($ipv eq "") {
$changeSet->add(["/sbin/conntrack",'-D','-s',$poolMember->{'IPAddress'}]);
}
if (defined($poolMember->{'IPNATInbound'}) && $poolMember->{'IPNATInbound'} eq "yes") {
my @ipComponents = split(/\//,$poolMember->{'IPAddress'});
my $dnatTo = $ipComponents[0];
if (@ipComponents < 2 || $ipComponents[1] eq "32" || $ipComponents[1] eq "128") {
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-A',"ots-dnat-$rxInterfaceID",
'-d',$poolMember->{'IPNATAddress'},
'-m','comment',
'--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'-j','DNAT',
'--to-destination', $dnatTo,
]);
} else {
$logger->log(LOG_WARN,"[TC] Cannot add inbound NAT for pool member '%s' [%s] with IP '%s', NAT '%s' (inbound: %s) to pool '%s' [%s]",
$poolMember->{'Username'},
$poolMember->{'ID'},
$poolMember->{'IPAddress'},
$poolMember->{'IPNATAddress'} // "",
$poolMember->{'IPNATInbound'} // "",
$pool->{'Name'},
$pool->{'ID'}
);
}
}
} }
# Post changeset # Post changeset
$kernel->post("_tc" => "queue" => $changeSet); $kernel->post("_tc" => "queue" => $changeSet);
...@@ -649,6 +899,7 @@ sub _session_poolmember_add ...@@ -649,6 +899,7 @@ sub _session_poolmember_add
} }
# Event handler for removing a pool member # Event handler for removing a pool member
sub _session_poolmember_remove sub _session_poolmember_remove
{ {
...@@ -667,112 +918,167 @@ sub _session_poolmember_remove ...@@ -667,112 +918,167 @@ sub _session_poolmember_remove
# Make sure its not NOTLIVE # Make sure its not NOTLIVE
if (getPoolMemberShaperState($pmid) & SHAPER_NOTLIVE) { if (getPoolMemberShaperState($pmid) & SHAPER_NOTLIVE) {
$logger->log(LOG_WARN,"[TC] Ignoring remove for pool member '%s' with IP '%s' [%s] from pool '%s'", $logger->log(LOG_WARN,"[TC] Ignoring remove for pool member '%s' with IP '%s', NAT '%s' (inbound: %s) [%s] from pool '%s'",
$poolMember->{'Username'}, $poolMember->{'Username'},
$poolMember->{'IPAddress'}, $poolMember->{'IPAddress'},
$poolMember->{'IPNATAddress'} // "",
$poolMember->{'IPNATInbound'} // "",
$poolMember->{'ID'}, $poolMember->{'ID'},
$pool->{'Name'} $pool->{'Name'}
); );
return; return;
} }
$logger->log(LOG_INFO,"[TC] Removing pool member '%s' with IP '%s' [%s] from pool '%s'", $logger->log(LOG_INFO,"[TC] Remove pool member '%s' [%s] with IP '%s', NAT '%s' (inbound: %s) from pool '%s' [%s]",
$poolMember->{'Username'}, $poolMember->{'Username'},
$poolMember->{'IPAddress'},
$poolMember->{'ID'}, $poolMember->{'ID'},
$pool->{'Name'} $poolMember->{'IPAddress'},
$poolMember->{'IPNATAddress'} // "",
$poolMember->{'IPNATInbound'} // "",
$pool->{'Name'},
$pool->{'ID'}
); );
# Grab our interfaces
my $txInterface = getPoolTxInterface($pool->{'ID'});
my $rxInterface = getPoolRxInterface($pool->{'ID'});
# Grab the filter ID's from the pool member which is linked to the traffic class
my $txFilter = getPoolMemberAttribute($poolMember->{'ID'},'tc.txfilter');
my $rxFilter = getPoolMemberAttribute($poolMember->{'ID'},'tc.rxfilter');
# Grab current class ID my $changeSet = TC::ChangeSet->new();
my $classID = getPoolAttribute($pool->{'ID'},'shaper.live.ClassID');
my $trafficPriority = getTrafficClassPriority($classID);
my $txInterfaceID = getPoolTxInterface($pool->{'ID'});
my $rxInterfaceID = getPoolRxInterface($pool->{'ID'});
my $changeSet = TC::ChangeSet->new(); my $txInterface = getInterface($txInterfaceID);
my $rxInterface = getInterface($rxInterfaceID);
# Clear up the filter my $rxPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.rxclass');
my $txPoolTcClass = getPoolAttribute($pool->{'ID'},'tc.txclass');
# Check what IP version we're dealing with
my $ipv = "";
if ( $poolMember->{'IPAddress'} =~ /:/ ) {
$ipv = 6;
}
# Remove traffic classification
$changeSet->add([ $changeSet->add([
'/sbin/tc','filter','del', "/sbin/ip${ipv}tables",
'dev',$txInterface, '-t','mangle',
'parent','1:', '-D',"ots-tcfor-$rxInterfaceID",
'prio',$trafficPriority, '-s',$poolMember->{'IPAddress'},
'handle',$txFilter, '-m','comment',
'protocol',$config->{'ip_protocol'}, '--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'u32', '-j','CLASSIFY',
'--set-class',"1:$rxPoolTcClass",
]); ]);
$changeSet->add([ $changeSet->add([
'/sbin/tc','filter','del', "/sbin/ip${ipv}tables",
'dev',$rxInterface, '-t','mangle',
'parent','1:', '-D',"ots-tcfor-$txInterfaceID",
'prio',$trafficPriority, '-d',$poolMember->{'IPAddress'},
'handle',$rxFilter, '-m','comment',
'protocol',$config->{'ip_protocol'}, '--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'u32', '-j','CLASSIFY',
'--set-class',"1:$txPoolTcClass",
]); ]);
# Remove NAT if we have any
if (defined($poolMember->{'IPNATAddress'}) && defined($poolMember->{'IPNATAddress'}) ne "") {
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-D',"ots-snat-$txInterfaceID",
'-s',$poolMember->{'IPAddress'},
'-m','comment',
'--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'-j','SNAT',
'--to-source', $poolMember->{'IPNATAddress'},
]);
if (defined($poolMember->{'IPNATInbound'}) && $poolMember->{'IPNATInbound'} eq "yes") {
my @ipComponents = split(/\//,$poolMember->{'IPAddress'});
my $dnatTo = $ipComponents[0];
if (@ipComponents < 2 || $ipComponents[1] eq "32" || $ipComponents[1] eq "128") {
$changeSet->add([
"/sbin/ip${ipv}tables",
'-t','nat',
'-D',"ots-dnat-$rxInterfaceID",
'-d',$poolMember->{'IPNATAddress'},
'-m','comment',
'--comment', "pool: ".$pool->{'Name'}.", member: ".$poolMember->{'Username'},
'-j','DNAT',
'--to-destination', $dnatTo,
]);
} else {
$logger->log(LOG_WARN,"[TC] Cannot remove inbound NAT for pool member '%s' [%s] with IP '%s', NAT '%s' (inbound: %s) to pool '%s' [%s]",
$poolMember->{'Username'},
$poolMember->{'ID'},
$poolMember->{'IPAddress'},
$poolMember->{'IPNATAddress'} // "",
$poolMember->{'IPNATInbound'} // "",
$pool->{'Name'},
$pool->{'ID'}
);
}
}
}
# Post changeset # Post changeset
$kernel->post("_tc" => "queue" => $changeSet); $kernel->post("_tc" => "queue" => $changeSet);
# Cleanup attributes
removePoolMemberAttribute($poolMember->{'ID'},'tc.txfilter');
removePoolMemberAttribute($poolMember->{'ID'},'tc.rxfilter');
# Mark as not live # Mark as not live
unsetPoolMemberShaperState($poolMember->{'ID'},SHAPER_LIVE|SHAPER_PENDING); unsetPoolMemberShaperState($poolMember->{'ID'},SHAPER_LIVE|SHAPER_PENDING);
setPoolMemberShaperState($poolMember->{'ID'},SHAPER_NOTLIVE); setPoolMemberShaperState($poolMember->{'ID'},SHAPER_NOTLIVE);
} }
# Grab pool ID from TC class # Grab pool ID from TC class
sub getPIDFromTcClass sub getPIDFromTcClass
{ {
my ($interface,$majorTcClass,$minorTcClass) = @_; my ($interfaceID,$majorTcClass,$minorTcClass) = @_;
# Return the pool ID if found # Return the pool ID if found
my $ref = __getRefByMinorTcClass($interface,$majorTcClass,$minorTcClass); my $ref = __getRefByMinorTcClass($interfaceID,$majorTcClass,$minorTcClass);
if (!defined($ref) || substr($ref,0,13) ne "_pool_class_:") { if (!defined($ref) || substr($ref,0,13) ne "_pool_class_:") {
return undef; return;
} }
return substr($ref,13); return substr($ref,13);
} }
# Function to return if this is linked to a pool's class # Function to return if this is linked to a pool's class
sub isPoolTcClass sub isPoolTcClass
{ {
my ($interface,$majorTcClass,$minorTcClass) = @_; my ($interfaceID,$majorTcClass,$minorTcClass) = @_;
my $pid = getPIDFromTcClass($interface,$majorTcClass,$minorTcClass); my $pid = getPIDFromTcClass($interfaceID,$majorTcClass,$minorTcClass);
if (!defined($pid)) { if (!defined($pid)) {
return undef; return;
} }
return $minorTcClass; return $minorTcClass;
} }
# Return the ClassID from a TC class # Return the ClassID from a TC class
# This is similar to isTcTrafficClassValid() but returns the ref, not the minor class # This is similar to isTcTrafficClassValid() but returns the ref, not the minor class
sub getCIDFromTcClass sub getCIDFromTcClass
{ {
my ($interface,$majorTcClass,$minorTcClass) = @_; my ($interfaceID,$majorTcClass,$minorTcClass) = @_;
# Grab ref # Grab ref
my $ref = __getRefByMinorTcClass($interface,$majorTcClass,$minorTcClass); my $ref = __getRefByMinorTcClass($interfaceID,$majorTcClass,$minorTcClass);
# If we're undefined return
if (!defined($ref)) {
return;
}
# If we're not a traffic class, just return # If we're not a traffic class, just return
if (substr($ref,0,16) ne "_traffic_class_:") { if (substr($ref,0,16) ne "_traffic_class_:") {
return undef; return;
} }
# Else return the part after the above tag # Else return the part after the above tag
...@@ -788,919 +1094,529 @@ sub getCIDFromTcClass ...@@ -788,919 +1094,529 @@ sub getCIDFromTcClass
# Function to initialize an interface # Function to initialize an interface
sub _tc_iface_init sub _tc_iface_init
{ {
my ($changeSet,$interface) = @_; my ($changeSet,$interfaceID) = @_;
# Grab our interface rate # Grab our interface rate
my $rate = getInterfaceRate($interface); my $interface = getInterface($interfaceID);
# Grab interface class configuration
my $trafficClasses = getInterfaceTrafficClasses($interface);
### --- Interface Setup
# Clear the qdisc from the interface # Clear the qdisc from the interface
$changeSet->add([ $changeSet->add([
'/sbin/tc','qdisc','del', '/sbin/tc','qdisc','del',
'dev',$interface, 'dev',$interface->{'Device'},
'root', 'root',
]); ]);
# Initialize the major TC class
my $interfaceTcClass = _reserveMajorTcClass($interfaceID,"root");
# Set interface RootClass
$globals->{'Interfaces'}->{$interfaceID} = {
'TcClass' => $interfaceTcClass
};
### --- Interface Traffic Class Setup
# Reserve our parent TC classes # Reserve our parent TC classes
foreach my $classID (sort {$a <=> $b} keys %{$trafficClasses}) { my @trafficClasses = getAllTrafficClasses();
# We don't really need the result, we just need the class created foreach my $trafficClassID (sort {$a <=> $b} @trafficClasses) {
_reserveTcClassByTrafficClassID($interface,$classID); # Record the class we get for this interface traffic class ID
my $interfaceTrafficClassTcClass = _reserveMinorTcClassByTrafficClassID($interfaceID,$trafficClassID);
} }
# Do we have a default pool? if so we must direct traffic there ### --- Interface Setup Part 2
my @qdiscOpts = ( );
my $defaultPool = getInterfaceDefaultPool($interface);
my $defaultPoolTcClass;
if (defined($defaultPool)) {
# Push unclassified traffic to this class
$defaultPoolTcClass = _getTcClassFromTrafficClassID($interface,$defaultPool);
push(@qdiscOpts,'default',$defaultPoolTcClass);
}
# Add root qdisc # Add root qdisc
$changeSet->add([ $changeSet->add([
'/sbin/tc','qdisc','add', '/sbin/tc','qdisc','add',
'dev',$interface, 'dev',$interface->{'Device'},
'root', 'root',
'handle','1:', 'handle','1:',
'htb', 'htb'
@qdiscOpts
]); ]);
# Attach our main rate to the qdisc # Attach our main limit on the qdisc
my $burst = int( ($interface->{'Limit'} / 8) * 1024 * 10); # Allow the entire interface to be emptied with a burst
$changeSet->add([ $changeSet->add([
'/sbin/tc','class','add', '/sbin/tc','class','add',
'dev',$interface, 'dev',$interface->{'Device'},
'parent','1:', 'parent','1:',
'classid','1:1', 'classid','1:1',
'htb', 'htb',
'rate',"${rate}kbit", 'rate',"$interface->{'Limit'}kbit",
'burst',"${rate}kb",
]); ]);
# Setup the classes # Class 0 is our interface, it points to 1 (the major TcClass)) : 1 (class below)
while ((my $classID, my $class) = each(%{$trafficClasses})) { $globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{'0'} = {
my $tcClass = _getTcClassFromTrafficClassID($interface,$classID); 'TcClass' => '1',
'CIR' => $interface->{'Limit'},
'Limit' => $interface->{'Limit'}
};
my $trafficPriority = getTrafficClassPriority($classID); ### --- Setup each class
# Setup the classes
foreach my $trafficClassID (@trafficClasses) {
my $interfaceTrafficClassID = isInterfaceTrafficClassValid($interfaceID,$trafficClassID);
my $interfaceTrafficClass = getEffectiveInterfaceTrafficClass2($interfaceTrafficClassID);
my $tcClass = _getTcClassFromTrafficClassID($interfaceID,$trafficClassID);
my $trafficPriority = getTrafficClassPriority($trafficClassID);
# Add class # Add class
$changeSet->add([ $changeSet->add([
'/sbin/tc','class','add', '/sbin/tc','class','add',
'dev',$interface, 'dev',$interface->{'Device'},
'parent','1:1', 'parent','1:1',
'classid',"1:$tcClass", 'classid',"1:$tcClass",
'htb', 'htb',
'rate',"$class->{'cir'}kbit", 'rate',"$interfaceTrafficClass->{'CIR'}kbit",
'ceil',"$class->{'limit'}kbit", 'ceil',"$interfaceTrafficClass->{'Limit'}kbit",
'prio',$trafficPriority, 'prio',$trafficPriority,
'burst', "$class->{'limit'}kb",
]); ]);
# Setup interface traffic class details
$globals->{'Interfaces'}->{$interfaceID}->{'TrafficClasses'}->{$trafficClassID} = {
'TcClass' => $tcClass,
'CIR' => $interfaceTrafficClass->{'CIR'},
'Limit' => $interfaceTrafficClass->{'Limit'}
};
} }
# Process our default pool traffic optimizations # Process our default pool traffic optimizations
my $defaultPool = getInterfaceDefaultPool($interfaceID);
if (defined($defaultPool)) { if (defined($defaultPool)) {
# If we have a rate for this iface, then use it my $interfaceTrafficClassID = isInterfaceTrafficClassValid($interfaceID,$defaultPool);
_tc_class_optimize($changeSet,$interface,$defaultPoolTcClass,$trafficClasses->{$defaultPool}->{'limit'}); my $interfaceTrafficClass = getEffectiveInterfaceTrafficClass2($interfaceTrafficClassID);
# Make the queue size big enough my $defaultPoolTcClass = _getTcClassFromTrafficClassID($interfaceID, $defaultPool);
my $queueSize = ($rate * 1024) / 8;
# Loop with IP versions
# RED metrics (sort of as per manpage) for my $ipv ("", "6") {
my $redAvPkt = 1000; $changeSet->add([
my $redMax = int($queueSize / 4); # 25% mark at 100% probabilty "/sbin/ip${ipv}tables",
my $redMin = int($redMax / 3); # Max/3 is when the probability starts '-t','mangle',
my $redBurst = int( ($redMin+$redMax) / (2*$redAvPkt)); '-A',"ots-tcfor-$interfaceID",
my $redLimit = $queueSize; '-m','comment',
'--comment', "Default",
my $prioTcClass = _getPrioTcClass($interface,$defaultPoolTcClass); '-j','CLASSIFY',
'--set-class',"1:$defaultPoolTcClass",
# Priority band ]);
my $prioBand = 1; }
$changeSet->add([
'/sbin/tc','qdisc','add',
'dev',$interface,
'parent',"$prioTcClass:".toHex($prioBand),
'handle',_reserveMajorTcClass($interface,"_default_pool_:$defaultPoolTcClass=>$prioBand").":",
'bfifo',
'limit',$queueSize,
]);
$prioBand++;
$changeSet->add([
'/sbin/tc','qdisc','add',
'dev',$interface,
'parent',"$prioTcClass:".toHex($prioBand),
'handle',_reserveMajorTcClass($interface,"_default_pool_:$defaultPoolTcClass=>$prioBand").":",
# TODO: NK - try enable the below
# 'estimator','1sec','4sec', # Quick monitoring, every 1s with 4s constraint
'red',
'min',$redMin,
'max',$redMax,
'limit',$redLimit,
'burst',$redBurst,
'avpkt',$redAvPkt,
# NK: ECN may cause excessive dips in traffic if there is an exceptional amount of traffic
# 'ecn'
# XXX: Very new kernels only ... use redflowlimit in future
# 'sfq',
# 'divisor','16384',
# 'headdrop',
# 'redflowlimit',$queueSize,
# 'ecn',
]);
$prioBand++; # If we have a rate for this iface, then use it
$changeSet->add([ _tc_class_optimize($changeSet,$interfaceID,$defaultPoolTcClass,$interfaceTrafficClass->{'Limit'});
'/sbin/tc','qdisc','add',
'dev',$interface,
'parent',"$prioTcClass:".toHex($prioBand),
'handle',_reserveMajorTcClass($interface,"_default_pool_:$defaultPoolTcClass=>$prioBand").":",
'red',
'min',$redMin,
'max',$redMax,
'limit',$redLimit,
'burst',$redBurst,
'avpkt',$redAvPkt,
# NK: ECN may cause excessive dips in traffic if there is an exceptional amount of traffic
# 'ecn'
]);
} }
} }
# Function to apply traffic optimizations to a classes # Function to apply traffic optimizations to a classes
# XXX: This probably needs working on # XXX: This probably needs working on
sub _tc_class_optimize sub _tc_class_optimize
{ {
my ($changeSet,$interface,$poolTcClass,$rate) = @_; my ($changeSet,$interfaceID,$poolTcClass,$rate) = @_;
# Rate for things like ICMP , ACK, SYN ... etc my $interface = getInterface($interfaceID);
my $rateBand1 = int($rate * (PROTO_RATE_LIMIT / 100)); my $prioTcClass = _reserveMajorTcClassByPrioClass($interfaceID,$poolTcClass);
$rateBand1 = PROTO_RATE_BURST_MIN if ($rateBand1 < PROTO_RATE_BURST_MIN);
my $rateBand1Burst = ($rateBand1 / 8) * PROTO_RATE_BURST_MAXM;
# Rate for things like VoIP/SSH/Telnet
my $rateBand2 = int($rate * (PRIO_RATE_LIMIT / 100));
$rateBand2 = PRIO_RATE_BURST_MIN if ($rateBand2 < PRIO_RATE_BURST_MIN);
my $rateBand2Burst = ($rateBand2 / 8) * PRIO_RATE_BURST_MAXM;
my $prioTcClass = _reserveMajorTcClassByPrioClass($interface,$poolTcClass); # Mbyte/s rate divided by average packet size to get packet limit, with a minimum of 127
my $sfqQueueLength = max(int((($rate / 8) * 1024) / 1000), 127);
# # Flows is Mbit/s divided by 20, with minimum of 127
# DEFINE 3 PRIO BANDS my $sfqFlows = max(int($rate / 1000 / 20) * 127, 127);
# my $sfqRedFlowLimit = ($sfqQueueLength / 4) * 1500; # Take 25% of the queue length and packet size of 1500
# We then prioritize traffic into 3 bands based on TOS # We then prioritize traffic into 3 bands based on TOS
$changeSet->add([ $changeSet->add([
'/sbin/tc','qdisc','add', '/sbin/tc','qdisc','add',
'dev',$interface, 'dev',$interface->{'Device'},
'parent',"1:$poolTcClass", 'parent',"1:$poolTcClass",
'handle',"$prioTcClass:", 'handle',"$prioTcClass:",
'prio', 'sfq',
'bands','3', 'limit', $sfqQueueLength, # first work out byte rate, then divide by average packet size of 1000
'priomap','2','2','2','2','2','2','2','2','2','2','2','2','2','2','2','2', 'flows', $sfqFlows,
'headdrop',
'perturb', 10,
'redflowlimit', $sfqRedFlowLimit,
'perturb', 10,
]); ]);
#
# CLASSIFICATIONS
#
# Prioritize ICMP up to a certain limit
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x1','0xff', # ICMP
'at',9+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand1}kbit",'burst',"${rateBand1Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# Prioritize ACK up to a certain limit
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u8','0x10','0xff', # ACK
'at',33+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand1}kbit",'burst',"${rateBand1Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# Prioritize SYN-ACK up to a certain limit
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u8','0x12','0xff', # SYN-ACK
'at',33+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand1}kbit",'burst',"${rateBand1Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# Prioritize FIN up to a certain limit
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u8','0x1','0xff', # FIN
'at',33+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand1}kbit",'burst',"${rateBand1Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# Prioritize RST up to a certain limit
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u8','0x4','0xff', # RST
'at',33+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand1}kbit",'burst',"${rateBand1Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# DNS
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x0035','0xffff', # SPORT 53
'at',20+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x0035','0xffff', # DPORT 53
'at',22+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# VOIP
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x13c4','0xffff', # SPORT 5060
'at',20+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x13c4','0xffff', # DPORT 5060
'at',22+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# SNMP
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0xa1','0xffff', # SPORT 161
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:1",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0xa1','0xffff', # DPORT 161
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:1",
]);
# TODO: Make this customizable not hard coded?
# Mikrotik Management Port
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x2063','0xffff', # SPORT 8291
'at',20+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u16','0x2063','0xffff', # DPORT 8291
'at',22+$config->{'iphdr_offset'},
'police',
'rate',"${rateBand2}kbit",'burst',"${rateBand2Burst}k",'continue',
'flowid',"$prioTcClass:1",
]);
# SMTP
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x19','0xffff', # SPORT 25
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x19','0xffff', # DPORT 25
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
# POP3
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x6e','0xffff', # SPORT 110
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x6e','0xffff', # DPORT 110
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
# IMAP
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x8f','0xffff', # SPORT 143
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x8f','0xffff', # DPORT 143
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
# HTTP
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x50','0xffff', # SPORT 80
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x50','0xffff', # DPORT 80
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
# HTTPS
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x1bb','0xffff', # SPORT 443
'at',20+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$prioTcClass:",
'prio','1',
'protocol',$config->{'ip_protocol'},
'u32',
'match','u8','0x6','0xff', # TCP
'at',9+$config->{'iphdr_offset'},
'match','u16','0x1bb','0xffff', # DPORT 443
'at',22+$config->{'iphdr_offset'},
'flowid',"$prioTcClass:2",
]);
} }
# Function to easily add a hash table
sub _tc_filter_add_dstlink
{
my ($changeSet,$interface,$parentID,$priority,$filterID,$protocol,$htHex,$ipHex,$cidr,$mask) = @_;
# Add hash table # Function to add a TC class
_tc_filter_hash_add($changeSet,$interface,$parentID,$priority,$filterID,$config->{'ip_protocol'}); sub _tc_class_add
# Add filter to it
_tc_filter_add($changeSet,$interface,$parentID,$priority,$filterID,$protocol,$htHex,$ipHex,"dst",16,$cidr,$mask);
}
# Function to easily add a hash table
sub _tc_filter_add_srclink
{ {
my ($changeSet,$interface,$parentID,$priority,$filterID,$protocol,$htHex,$ipHex,$cidr,$mask) = @_; my ($changeSet,$interfaceID,$majorTcClass,$trafficClassTcClass,$poolTcClass,$rate,$ceil,$trafficPriority) = @_;
# Add hash table my $interface = getInterface($interfaceID);
_tc_filter_hash_add($changeSet,$interface,$parentID,$priority,$filterID,$config->{'ip_protocol'});
# Add filter to it
_tc_filter_add($changeSet,$interface,$parentID,$priority,$filterID,$protocol,$htHex,$ipHex,"src",12,$cidr,$mask);
}
# # Set burst to a sane value, in this case the CIR (or $rate) size
# my $burst = int($rate / 8);
# my $cburst = int(($ceil - $rate) / 8 / 10);
# # NK: cburst should not exceed burst, if it does, just use the burst value
# # this ensures we do not get negative burst
# if ($cburst > $burst) {
# $cburst = $burst;
# }
my $burst = int( ($ceil / 8) * 1024 * 10);
my $cburst = int($rate / 8 / 5) + 2; # +2kb to cover 1600
my $quantum = int($cburst / 2) * 1024; # Burst in bytes
# Function to easily add a hash table # Create main rate limiting classes
sub _tc_filter_add_flowlink
{
my ($changeSet,$interface,$parentID,$priority,$protocol,$htHex,$ipHex,$type,$offset,$ip,$poolTcClass) = @_;
# Link hash table
$changeSet->add([ $changeSet->add([
'/sbin/tc','filter','add', '/sbin/tc','class','add',
'dev',$interface, 'dev',$interface->{'Device'},
'parent',"$parentID:", 'parent',"$majorTcClass:$trafficClassTcClass",
'prio',$priority, 'classid',"$majorTcClass:$poolTcClass",
'handle',"$htHex:$ipHex:1", 'htb',
'protocol',$protocol, 'rate', "${rate}kbit",
'u32', 'ceil', "${ceil}kbit",
# Root hash table # 'prio', $trafficPriority,
'ht',"$htHex:$ipHex:", # 'burst', "${burst}kb",
'match','ip',$type,$ip, # 'cburst', "${cburst}kb",
'at',$offset+$config->{'iphdr_offset'},
# Link to our flow
'flowid',"1:$poolTcClass",
]); ]);
} }
# Function to easily add a hash table
sub _tc_filter_hash_add # Function to change a TC class
sub _tc_class_change
{ {
my ($changeSet,$interface,$parentID,$priority,$filterID,$protocol) = @_; my ($changeSet,$interfaceID,$majorTcClass,$trafficClassTcClass,$poolTcClass,$rate,$ceil,$trafficPriority) = @_;
# Create second level hash table for $ip1 my $interface = getInterface($interfaceID);
$changeSet->add([
'/sbin/tc','filter','add',
'dev',$interface,
'parent',"$parentID:",
'prio',$priority,
'handle',"$filterID:",
'protocol',$protocol,
'u32',
'divisor','256',
]);
}
my @args = ();
# Function to easily add a hash table # If ceil is not available, set it to the CIR (or $rate in this case)
sub _tc_filter_add if (!defined($ceil)) {
{ $ceil = $rate;
my ($changeSet,$interface,$parentID,$priority,$filterID,$protocol,$htHex,$ipHex,$type,$offset,$cidr,$mask) = @_; }
# Link hash table # # Set the burst rate to the CIR (or $rate in this case)
# my $burst = int($rate / 8);
# my $cburst = int(($ceil - $rate) / 8 / 10);
# # NK: cburst should not exceed burst, if it does, just use the burst value
# # this ensures we do not get negative burst
# if ($cburst > $burst) {
# $cburst = $burst;
# }
my $burst = int( ($ceil / 8) * 1024 * 10);
my $cburst = int($rate / 8 / 5) + 2; # +2kb to cover 1600
my $quantum = int($cburst / 2) * 1024; # Burst in bytes
# # Check if we have a priority
# if (defined($trafficPriority)) {
# push(@args,'prio',$trafficPriority);
# }
# Create main rate limiting classes
$changeSet->add([ $changeSet->add([
'/sbin/tc','filter','add', '/sbin/tc','class','change',
'dev',$interface, 'dev',$interface->{'Device'},
'parent',"$parentID:", 'parent',"$majorTcClass:$trafficClassTcClass",
'prio',$priority, 'classid',"$majorTcClass:$poolTcClass",
'protocol',$protocol, 'htb',
'u32', 'rate', "${rate}kbit",
# Root hash table 'ceil', "${ceil}kbit",
'ht',"$htHex:$ipHex:", # 'burst', "${burst}kb",
'match','ip',$type,$cidr, # 'cburst', "${cburst}kb",
'at',$offset+$config->{'iphdr_offset'}, @args
'hashkey','mask',"0x$mask",
'at',$offset+$config->{'iphdr_offset'},
# Link to our hash table
'link',"$filterID:"
]); ]);
} }
# Function to add a TC class
sub _tc_class_add
{
my ($changeSet,$interface,$majorTcClass,$trafficClassTcClass,$poolTcClass,$rate,$ceil,$trafficPriority) = @_;
# Set burst to a sane value
my $burst = int($ceil / 8 / 5);
# Create main rate limiting classes
$changeSet->add([
'/sbin/tc','class','add',
'dev',$interface,
'parent',"$majorTcClass:$trafficClassTcClass",
'classid',"$majorTcClass:$poolTcClass",
'htb',
'rate', "${rate}kbit",
'ceil', "${ceil}kbit",
'prio', $trafficPriority,
'burst', "${burst}kb",
]);
}
# Function to change a TC class
sub _tc_class_change
{
my ($changeSet,$interface,$majorTcClass,$trafficClassTcClass,$poolTcClass,$rate,$ceil,$trafficPriority) = @_;
# Set burst to a sane value
my $burst = int($ceil / 8 / 5);
# Create main rate limiting classes
$changeSet->add([
'/sbin/tc','class','change',
'dev',$interface,
'parent',"$majorTcClass:$trafficClassTcClass",
'classid',"$majorTcClass:$poolTcClass",
'htb',
'rate', "${rate}kbit",
'ceil', "${ceil}kbit",
'prio', $trafficPriority,
'burst', "${burst}kb",
]);
}
# Get a pool TC class from pool ID # Get a pool TC class from pool ID
sub _reserveTcClassByPoolID sub _reserveMinorTcClassByPoolID
{ {
my ($interface,$pid) = @_; my ($interfaceID,$pid) = @_;
return __reserveMinorTcClass($interface,TC_ROOT_CLASS,"_pool_class_:$pid"); return __reserveMinorTcClass($interfaceID,TC_ROOT_CLASS,"_pool_class_:$pid");
} }
# Get a traffic class TC class # Get a traffic class TC class
sub _reserveTcClassByTrafficClassID sub _reserveMinorTcClassByTrafficClassID
{ {
my ($interface,$classID) = @_; my ($interfaceID,$trafficClassID) = @_;
return __reserveMinorTcClass($interface,TC_ROOT_CLASS,"_traffic_class_:$classID"); return __reserveMinorTcClass($interfaceID,TC_ROOT_CLASS,"_traffic_class_:$trafficClassID");
} }
# Get a prio class TC class # Get a prio class TC class
# This is a MAJOR class! # This is a MAJOR class!
sub _reserveMajorTcClassByPrioClass sub _reserveMajorTcClassByPrioClass
{ {
my ($interface,$classID) = @_; my ($interfaceID,$trafficClassID) = @_;
return _reserveMajorTcClass($interface,"_priority_class_:$classID"); return _reserveMajorTcClass($interfaceID,"_priority_class_:$trafficClassID");
} }
# Return TC class from a traffic class ID # Return TC class from a traffic class ID
sub _getTcClassFromTrafficClassID sub _getTcClassFromTrafficClassID
{ {
my ($interface,$classID) = @_; my ($interfaceID,$trafficClassID) = @_;
return __getMinorTcClassByRef($interface,TC_ROOT_CLASS,"_traffic_class_:$classID"); return __getMinorTcClassByRef($interfaceID,TC_ROOT_CLASS,"_traffic_class_:$trafficClassID");
} }
# Return prio TC class using class # Return prio TC class using class
# This returns a MAJOR class from a tc class # This returns a MAJOR class from a tc class
sub _getPrioTcClass sub _getPrioTcClass
{ {
my ($interface,$tcClass) = @_; my ($interfaceID,$tcClass) = @_;
return __getMajorTcClassByRef($interface,"_priority_class_:$tcClass"); return __getMajorTcClassByRef($interfaceID,"_priority_class_:$tcClass");
} }
# Function to dispose of a TC class # Function to dispose of a TC class
sub _disposePoolTcClass sub _disposePoolTcClass
{ {
my ($interface,$tcClass) = @_; my ($interfaceID,$tcClass) = @_;
return __disposeMinorTcClass($interface,TC_ROOT_CLASS,$tcClass); return __disposeMinorTcClass($interfaceID,TC_ROOT_CLASS,$tcClass);
} }
# Function to dispose of a major TC class # Function to dispose of a major TC class
# Uses a TC class to get a MAJOR class, then disposes it # Uses a TC class to get a MAJOR class, then disposes it
sub _disposePrioTcClass sub _disposePrioTcClass
{ {
my ($interface,$tcClass) = @_; my ($interfaceID,$tcClass) = @_;
# If we can grab the major class dipose of it # If we can grab the major class dipose of it
my $majorTcClass = _getPrioTcClass($interface,$tcClass); my $majorTcClass = _getPrioTcClass($interfaceID,$tcClass);
if (!defined($majorTcClass)) { if (!defined($majorTcClass)) {
return undef; return;
} }
return __disposeMajorTcClass($interface,$majorTcClass); return __disposeMajorTcClass($interfaceID,$majorTcClass);
} }
# Function to get next available TC class # Function to get next available TC class
sub __reserveMinorTcClass sub __reserveMinorTcClass
{ {
my ($interface,$majorTcClass,$ref) = @_; my ($interfaceID,$majorTcClass,$ref) = @_;
# Setup defaults if we don't have anything defined # Setup defaults if we don't have anything defined
if (!defined($tcClasses->{$interface}) || !defined($tcClasses->{$interface}->{$majorTcClass})) { if (!defined($globals->{'TcClasses'}->{$interfaceID}) || !defined($globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass})) {
$tcClasses->{$interface}->{$majorTcClass} = { $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass} = {
'free' => [ ], # Skip 0 and 1
'track' => { }, 'Counter' => 2,
'reverse' => { }, 'Free' => [ ],
'Track' => { },
'Reverse' => { },
}; };
} }
# Maybe we have one free? # Maybe we have one free?
my $minorTcClass = shift(@{$tcClasses->{$interface}->{$majorTcClass}->{'free'}}); my $minorTcClass = shift(@{$globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Free'}});
# Generate new number # Generate new number
if (!$minorTcClass) { if (!$minorTcClass) {
$minorTcClass = keys %{$tcClasses->{$interface}->{$majorTcClass}->{'track'}}; $minorTcClass = $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Counter'}++;
$minorTcClass += 2; # Skip 0 and 1
# Hex it # Hex it
$minorTcClass = toHex($minorTcClass); $minorTcClass = toHex($minorTcClass);
} }
$tcClasses->{$interface}->{$majorTcClass}->{'track'}->{$minorTcClass} = $ref; $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Track'}->{$minorTcClass} = $ref;
$tcClasses->{$interface}->{$majorTcClass}->{'reverse'}->{$ref} = $minorTcClass; $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Reverse'}->{$ref} = $minorTcClass;
return $minorTcClass; return $minorTcClass;
} }
# Function to get next available major TC class # Function to get next available major TC class
sub _reserveMajorTcClass sub _reserveMajorTcClass
{ {
my ($interface,$ref) = @_; my ($interfaceID,$ref) = @_;
# Setup defaults if we don't have anything defined # Setup defaults if we don't have anything defined
if (!defined($tcClasses->{$interface})) { if (!defined($globals->{'TcClasses'}->{$interfaceID})) {
$tcClasses->{$interface} = { $globals->{'TcClasses'}->{$interfaceID} = {
'free' => [ ], # Skip 0
'track' => { }, 'Counter' => 1,
'reverse' => { }, 'Free' => [ ],
'Track' => { },
'Reverse' => { },
}; };
} }
# Maybe we have one free? # Maybe we have one free?
my $majorTcClass = shift(@{$tcClasses->{$interface}->{'free'}}); my $majorTcClass = shift(@{$globals->{'TcClasses'}->{$interfaceID}->{'Free'}});
# Generate new number # Generate new number
if (!$majorTcClass) { if (!$majorTcClass) {
$majorTcClass = keys %{$tcClasses->{$interface}->{'track'}}; $majorTcClass = $globals->{'TcClasses'}->{$interfaceID}->{'Counter'}++;
$majorTcClass += 2; # Skip 0 and 1
# Hex it # Hex it
$majorTcClass = toHex($majorTcClass); $majorTcClass = toHex($majorTcClass);
} }
$tcClasses->{$interface}->{'track'}->{$majorTcClass} = $ref; $globals->{'TcClasses'}->{$interfaceID}->{'Track'}->{$majorTcClass} = $ref;
$tcClasses->{$interface}->{'reverse'}->{$ref} = $majorTcClass; $globals->{'TcClasses'}->{$interfaceID}->{'Reverse'}->{$ref} = $majorTcClass;
return $majorTcClass; return $majorTcClass;
} }
# Get a minor class by its rerf # Get a minor class by its rerf
sub __getMinorTcClassByRef sub __getMinorTcClassByRef
{ {
my ($interface,$majorTcClass,$ref) = @_; my ($interfaceID,$majorTcClass,$ref) = @_;
if (!defined($tcClasses->{$interface}) || !defined($tcClasses->{$interface}->{$majorTcClass})) { if (!defined($globals->{'TcClasses'}->{$interfaceID}) || !defined($globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass})) {
return undef; return;
} }
return $tcClasses->{$interface}->{$majorTcClass}->{'reverse'}->{$ref}; return $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Reverse'}->{$ref};
} }
# Get a major class by its rerf # Get a major class by its rerf
sub __getMajorTcClassByRef sub __getMajorTcClassByRef
{ {
my ($interface,$ref) = @_; my ($interfaceID,$ref) = @_;
if (!defined($tcClasses->{$interface})) { if (!defined($globals->{'TcClasses'}->{$interfaceID})) {
return undef; return;
} }
return $tcClasses->{$interface}->{'reverse'}->{$ref}; return $globals->{'TcClasses'}->{$interfaceID}->{'Reverse'}->{$ref};
} }
# Get ref using the minor tc class # Get ref using the minor tc class
sub __getRefByMinorTcClass sub __getRefByMinorTcClass
{ {
my ($interface,$majorTcClass,$minorTcClass) = @_; my ($interfaceID,$majorTcClass,$minorTcClass) = @_;
if (!defined($tcClasses->{$interface}) || !defined($tcClasses->{$interface}->{$majorTcClass})) { if (!defined($globals->{'TcClasses'}->{$interfaceID}) || !defined($globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass})) {
return undef; return;
} }
return $tcClasses->{$interface}->{$majorTcClass}->{'track'}->{$minorTcClass}; return $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Track'}->{$minorTcClass};
} }
# Function to dispose of a TC class # Function to dispose of a TC class
sub __disposeMinorTcClass sub __disposeMinorTcClass
{ {
my ($interface,$majorTcClass,$tcMinorClass) = @_; my ($interfaceID,$majorTcClass,$tcMinorClass) = @_;
my $ref = $tcClasses->{$interface}->{$majorTcClass}->{'track'}->{$tcMinorClass}; my $ref = $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Track'}->{$tcMinorClass};
# Push onto free list # Push onto free list
push(@{$tcClasses->{$interface}->{$majorTcClass}->{'free'}},$tcMinorClass); push(@{$globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Free'}},$tcMinorClass);
# Blank the value # Blank the value
$tcClasses->{$interface}->{$majorTcClass}->{'track'}->{$tcMinorClass} = undef; $globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Track'}->{$tcMinorClass} = undef;
delete($tcClasses->{$interface}->{$majorTcClass}->{'reverse'}->{$ref}); delete($globals->{'TcClasses'}->{$interfaceID}->{$majorTcClass}->{'Reverse'}->{$ref});
} }
# Function to dispose of a major TC class # Function to dispose of a major TC class
sub __disposeMajorTcClass sub __disposeMajorTcClass
{ {
my ($interface,$tcMajorClass) = @_; my ($interfaceID,$tcMajorClass) = @_;
my $ref = $tcClasses->{$interface}->{'track'}->{$tcMajorClass}; my $ref = $globals->{'TcClasses'}->{$interfaceID}->{'Track'}->{$tcMajorClass};
# Push onto free list # Push onto free list
push(@{$tcClasses->{$interface}->{'free'}},$tcMajorClass); push(@{$globals->{'TcClasses'}->{$interfaceID}->{'Free'}},$tcMajorClass);
# Blank the value # Blank the value
$tcClasses->{$interface}->{'track'}->{$tcMajorClass} = undef; $globals->{'TcClasses'}->{$interfaceID}->{'Track'}->{$tcMajorClass} = undef;
delete($tcClasses->{$interface}->{'reverse'}->{$ref}); delete($globals->{'TcClasses'}->{$interfaceID}->{'Reverse'}->{$ref});
} }
# Function to get next available TC filter # Function to get next available TC filter
sub _reserveTcFilter sub _reserveTcFilter
{ {
my ($interface,$ref) = @_; my ($interfaceID,$ref) = @_;
# Setup defaults if we don't have anything defined # Setup defaults if we don't have anything defined
if (!defined($tcFilters->{$interface})) { if (!defined($globals->{'TcFilters'}->{$interfaceID})) {
$tcFilters->{$interface} = { $globals->{'TcFilters'}->{$interfaceID} = {
'free' => [ ], # Skip 0 and 1
'track' => { }, 'Counter' => 2,
'Free' => [ ],
'Track' => { },
}; };
} }
# Maybe we have one free? # Maybe we have one free?
my $filterID = shift(@{$tcFilters->{$interface}->{'free'}}); my $filterID = shift(@{$globals->{'TcFilters'}->{$interfaceID}->{'Free'}});
# Generate new number # Generate new number
if (!$filterID) { if (!$filterID) {
$filterID = keys %{$tcFilters->{$interface}->{'track'}}; $filterID = $globals->{'TcFilters'}->{$interfaceID}->{'Counter'}++;
# Bump ID
$filterID += 2; # Skip 0 and 1
# We cannot use ID 800, its internal # We cannot use ID 800, its internal
$filterID = 801 if ($filterID == 800); $filterID = $globals->{'TcFilters'}->{$interfaceID}->{'Counter'}++ if ($filterID == 800);
# Hex it # Hex it
$filterID = toHex($filterID); $filterID = toHex($filterID);
} }
$tcFilters->{$interface}->{'track'}->{$filterID} = $ref; $globals->{'TcFilters'}->{$interfaceID}->{'Track'}->{$filterID} = $ref;
return $filterID; return $filterID;
} }
# Function to dispose of a TC Filter # Function to dispose of a TC Filter
sub _disposeTcFilter sub _disposeTcFilter
{ {
my ($interface,$filterID) = @_; my ($interfaceID,$filterID) = @_;
# Push onto free list # Push onto free list
push(@{$tcFilters->{$interface}->{'free'}},$filterID); push(@{$globals->{'TcFilters'}->{$interfaceID}->{'Free'}},$filterID);
# Blank the value # Blank the value
$tcFilters->{$interface}->{'track'}->{$filterID} = undef; $globals->{'TcFilters'}->{$interfaceID}->{'Track'}->{$filterID} = undef;
} }
# #
# Task/child communication & handling stuff # Task/child communication & handling stuff
# #
# Initialize our tc session # Initialize our tc session
sub _task_session_start sub _task_session_start
{ {
...@@ -1717,6 +1633,7 @@ sub _task_session_start ...@@ -1717,6 +1633,7 @@ sub _task_session_start
} }
# Add task to queue # Add task to queue
sub _task_add_to_queue sub _task_add_to_queue
{ {
...@@ -1726,11 +1643,7 @@ sub _task_add_to_queue ...@@ -1726,11 +1643,7 @@ sub _task_add_to_queue
# Extract the changeset into commands # Extract the changeset into commands
my $numChanges = 0; my $numChanges = 0;
foreach my $cmd ($changeSet->extract()) { foreach my $cmd ($changeSet->extract()) {
# Rip off path to tc command push(@{$globals->{'TaskQueue'}},$cmd);
shift(@{$cmd});
# Build commandline string
my $cmdStr = join(' ',@{$cmd});
push(@taskQueue,$cmdStr);
$numChanges++; $numChanges++;
} }
...@@ -1738,27 +1651,6 @@ sub _task_add_to_queue ...@@ -1738,27 +1651,6 @@ sub _task_add_to_queue
} }
# Send the next command in the task direction
sub _task_put_next
{
my ($heap,$task) = @_;
# Task was busy, this signifies its done, so lets take the next command
if (my $cmdStr = shift(@taskQueue)) {
# Remove off idle task list if its there
delete($heap->{'idle_tasks'}->{$task->ID});
$task->put($cmdStr);
$logger->log(LOG_DEBUG,"[TC] TASK/%s: Starting '%s' as %s with PID %s",$task->ID,$cmdStr,$task->ID,$task->PID);
# If there is no commands in the queue, set it to idle
} else {
# Set task to idle
$heap->{'idle_tasks'}->{$task->ID} = $task;
}
}
# Queue a task # Queue a task
sub _task_queue sub _task_queue
...@@ -1770,63 +1662,57 @@ sub _task_queue ...@@ -1770,63 +1662,57 @@ sub _task_queue
_task_add_to_queue($changeSet); _task_add_to_queue($changeSet);
# Trigger a run if list is not empty # Trigger a run if list is not empty
if (@taskQueue) { if (@{$globals->{'TaskQueue'}}) {
$kernel->yield("_task_run_next"); $kernel->yield("_task_run_next");
} }
} }
# Run next task # Run next task
sub _task_run_next sub _task_run_next
{ {
my ($kernel,$heap) = @_[KERNEL,HEAP]; my ($kernel,$heap) = @_[KERNEL,HEAP];
# If we already have children processing tasks, don't create another
if (keys %{$heap->{'task_by_wid'}}) { if (keys %{$heap->{'task_by_wid'}}) {
# Loop with idle tasks ... return if we found one # NK: Limit concurrency to 1
foreach my $task_id (keys %{$heap->{'idle_tasks'}}) {
_task_put_next($heap,$heap->{'idle_tasks'}->{$task_id});
# XXX: Limit concurrency to 1
last;
}
# XXX: Limit concurrency to 1
return; return;
} }
# Check if we have a task coming off the top of the task queue # Check if we have a task coming off the top of the task queue
if (@taskQueue) { if (my $cmd = shift(@{$globals->{'TaskQueue'}})) {
my $cmdStr = encode_json($cmd);
# Create task # Create task
my $task = POE::Wheel::Run->new( my $task = POE::Wheel::Run->new(
Program => [ '/sbin/tc', '-force', '-batch' ], Program => $cmd,
Conduit => 'pipe', StdoutFilter => POE::Filter::Line->new( Literal => "\n" ),
StdioFilter => POE::Filter::Line->new( Literal => "\n" ),
StderrFilter => POE::Filter::Line->new( Literal => "\n" ), StderrFilter => POE::Filter::Line->new( Literal => "\n" ),
StdoutEvent => '_task_child_stdout', StdoutEvent => '_task_child_stdout',
StderrEvent => '_task_child_stderr', StderrEvent => '_task_child_stderr',
CloseEvent => '_task_child_close', CloseEvent => '_task_child_close',
StdinEvent => '_task_child_stdin', # ErrorEvent => '_task_child_error',
ErrorEvent => '_task_child_error', ) or $logger->log(LOG_ERR,"[TC] TASK: Unable to start task: $cmdStr");
) or $logger->log(LOG_ERR,"[TC] TASK: Unable to start task");
# Set task ID # Set task ID
my $task_id = $task->ID; my $task_id = $task->ID;
# Intercept SIGCHLD # Intercept SIGCHLD
$kernel->sig_child($task->PID, "_SIGCHLD"); $kernel->sig_child($task->PID, "_SIGCHLD");
# Wheel events include the wheel's ID. # Wheel events include the wheel's ID.
$heap->{'task_by_wid'}->{$task_id} = $task; $heap->{'task_by_wid'}->{$task_id} = $task;
# Signal events include the process ID. # Signal events include the process ID.
$heap->{'task_by_pid'}->{$task_id} = $task; $heap->{'task_by_pid'}->{$task->PID} = $task;
_task_put_next($heap,$task); # Build commandline string
$logger->log( LOG_DEBUG, "[TC] TASK/%s: Starting '%s' as %s with PID %s", $task->ID, $cmdStr, $task->ID, $task->PID );
} }
} }
# Child writes to STDOUT # Child writes to STDOUT
sub _task_child_stdout sub _task_child_stdout
{ {
...@@ -1839,6 +1725,7 @@ sub _task_child_stdout ...@@ -1839,6 +1725,7 @@ sub _task_child_stdout
} }
# Child writes to STDERR # Child writes to STDERR
sub _task_child_stderr sub _task_child_stderr
{ {
...@@ -1847,21 +1734,7 @@ sub _task_child_stderr ...@@ -1847,21 +1734,7 @@ sub _task_child_stderr
my $task = $heap->{'task_by_wid'}->{$task_id}; my $task = $heap->{'task_by_wid'}->{$task_id};
$logger->log(LOG_WARN,"[TC] TASK/%s: STDOUT => %s",$task_id,$stdout); $logger->log(LOG_WARN,"[TC] TASK/%s: STDERR => %s",$task_id,$stdout);
}
# Child flushed to STDIN
sub _task_child_stdin
{
my ($kernel,$heap,$task_id) = @_[KERNEL,HEAP,ARG0];
my $task = $heap->{'task_by_wid'}->{$task_id};
$logger->log(LOG_DEBUG,"[TC] TASK/%s is READY",$task_id);
# And shove another queued command its direction
_task_put_next($heap,$task);
} }
...@@ -1885,15 +1758,15 @@ sub _task_child_close ...@@ -1885,15 +1758,15 @@ sub _task_child_close
# Remove other references # Remove other references
delete($heap->{'task_by_wid'}->{$task_id}); delete($heap->{'task_by_wid'}->{$task_id});
delete($heap->{'task_by_pid'}->{$task->PID}); delete($heap->{'task_by_pid'}->{$task->PID});
delete($heap->{'idle_tasks'}->{$task_id});
# Start next one, if there is a next one # Start next one, if there is a next one
if (@taskQueue) { if (@{$globals->{'TaskQueue'}}) {
$kernel->yield("_task_run_next"); $kernel->yield("_task_run_next");
} }
} }
# Child got an error event, lets remove it too # Child got an error event, lets remove it too
sub _task_child_error sub _task_child_error
{ {
...@@ -1914,15 +1787,15 @@ sub _task_child_error ...@@ -1914,15 +1787,15 @@ sub _task_child_error
# Remove other references # Remove other references
delete($heap->{'task_by_wid'}->{$task_id}); delete($heap->{'task_by_wid'}->{$task_id});
delete($heap->{'task_by_pid'}->{$task->PID}); delete($heap->{'task_by_pid'}->{$task->PID});
delete($heap->{'idle_tasks'}->{$task_id});
# Start next one, if there is a next one # Start next one, if there is a next one
if (@taskQueue) { if (@{$globals->{'TaskQueue'}}) {
$kernel->yield("_task_run_next"); $kernel->yield("_task_run_next");
} }
} }
# Reap the dead child # Reap the dead child
sub _task_SIGCHLD sub _task_SIGCHLD
{ {
...@@ -1939,10 +1812,10 @@ sub _task_SIGCHLD ...@@ -1939,10 +1812,10 @@ sub _task_SIGCHLD
# Remove other references # Remove other references
delete($heap->{'task_by_wid'}->{$task->ID}); delete($heap->{'task_by_wid'}->{$task->ID});
delete($heap->{'task_by_pid'}->{$pid}); delete($heap->{'task_by_pid'}->{$pid});
delete($heap->{'idle_tasks'}->{$task->ID});
} }
# Handle SIGINT # Handle SIGINT
sub _task_SIGINT sub _task_SIGINT
{ {
...@@ -1982,6 +1855,7 @@ sub new ...@@ -1982,6 +1855,7 @@ sub new
} }
# Add a change to the list # Add a change to the list
sub add sub add
{ {
...@@ -1991,6 +1865,7 @@ sub add ...@@ -1991,6 +1865,7 @@ sub add
} }
# Return the list # Return the list
sub extract sub extract
{ {
...@@ -2001,5 +1876,20 @@ sub extract ...@@ -2001,5 +1876,20 @@ sub extract
# Return the list
sub debug
{
my $self = shift;
my @debug = ();
foreach my $item ($self->extract) {
push(@debug,join(' ',@{$item}));
}
return @debug;
}
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper Linux tcstats traffic shaping statistics # OpenTrafficShaper Linux tcstats traffic shaping statistics
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -24,13 +24,13 @@ use warnings; ...@@ -24,13 +24,13 @@ use warnings;
use POE qw( Wheel::Run Filter::Line ); use POE qw( Wheel::Run Filter::Line );
use POE::Filter::TCStatistics; use opentrafficshaper::POE::Filter::TCStatistics;
use opentrafficshaper::constants; use opentrafficshaper::constants;
use opentrafficshaper::logger; use opentrafficshaper::logger;
use opentrafficshaper::utils;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
getInterface
getInterfaces getInterfaces
); );
...@@ -45,9 +45,9 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -45,9 +45,9 @@ our (@ISA,@EXPORT,@EXPORT_OK);
); );
use constant { use constant {
VERSION => '0.1.2', VERSION => '1.0.0',
# How often our config check ticks # How often we tick
TICK_PERIOD => 5, TICK_PERIOD => 5,
}; };
...@@ -64,28 +64,34 @@ our $pluginInfo = { ...@@ -64,28 +64,34 @@ our $pluginInfo = {
}; };
# Copy of system globals # Our globals
my $globals; my $globals;
# Copy of system logger
my $logger; my $logger;
# Last stats pulls # Last stats pulls
my $lastStats = { }; #
# $globals->{'LastStats'}
# Initialize plugin # Initialize plugin
sub plugin_init sub plugin_init
{ {
$globals = shift; my $system = shift;
# Setup our environment # Setup our environment
$logger = $globals->{'logger'}; $logger = $system->{'logger'};
$logger->log(LOG_NOTICE,"[TCSTATS] OpenTrafficShaper tc Statistics Integration v%s - Copyright (c) 2013-2014, AllWorldIT", $logger->log(LOG_NOTICE,"[TCSTATS] OpenTrafficShaper tc Statistics Integration v%s - Copyright (c) 2013-2014, AllWorldIT",
VERSION VERSION
); );
# Initialize
$globals->{'LastStats'} = { };
# This session is our main session, its alias is "shaper" # This session is our main session, its alias is "shaper"
POE::Session->create( POE::Session->create(
inline_states => { inline_states => {
...@@ -106,13 +112,25 @@ sub plugin_init ...@@ -106,13 +112,25 @@ sub plugin_init
} }
# Start the plugin # Start the plugin
sub plugin_start sub plugin_start
{ {
my @interfaces = getInterfaces();
my $now = time();
# Initialize last stats
foreach my $interfaceID (@interfaces) {
$globals->{'LastStats'}->{$interfaceID} = $now;
}
$logger->log(LOG_INFO,"[TCSTATS] Started"); $logger->log(LOG_INFO,"[TCSTATS] Started");
} }
# Initialize this plugins main POE session # Initialize this plugins main POE session
sub _session_start sub _session_start
{ {
...@@ -129,6 +147,7 @@ sub _session_start ...@@ -129,6 +147,7 @@ sub _session_start
} }
# Shut down session # Shut down session
sub _session_stop sub _session_stop
{ {
...@@ -145,6 +164,7 @@ sub _session_stop ...@@ -145,6 +164,7 @@ sub _session_stop
} }
# Time ticker for processing changes # Time ticker for processing changes
sub _session_tick sub _session_tick
{ {
...@@ -154,27 +174,26 @@ sub _session_tick ...@@ -154,27 +174,26 @@ sub _session_tick
# Now # Now
my $now = time(); my $now = time();
# Loop with interfaces that need stats # Get sorted list of interfaces
my $interfaceCount = 0; my @interfaces = sort { $globals->{'LastStats'}->{$a} <=> $globals->{'LastStats'}->{$b} } getInterfaces();
foreach my $interface (@{getInterfaces()}) {
# Skip to next if we've already run for this interface # Grab the first interface in the list to process
if (defined($lastStats->{$interface}) && my $interfaceID = shift(@interfaces);
$lastStats->{$interface} + opentrafficshaper::plugins::statistics::STATISTICS_PERIOD > $now
) { # Check if its old enough to process stats for
next; if ($now - $globals->{'LastStats'}->{$interfaceID} > opentrafficshaper::plugins::statistics::STATISTICS_PERIOD) {
} my $interface = getInterface($interfaceID);
$logger->log(LOG_INFO,"[TCSTATS] Generating stats for '%s'",$interface); $logger->log(LOG_INFO,"[TCSTATS] Generating stats for '%s'",$interfaceID);
# TC commands to run # TC commands to run
my $cmd = [ '/sbin/tc', '-s', 'class', 'show', 'dev', $interface, 'parent', '1:' ]; my $cmd = [ '/sbin/tc', '-j', '-s', 'class', 'show', 'dev', $interface->{'Device'}, 'parent', '1:' ];
# Create task # Create task
my $task = POE::Wheel::Run->new( my $task = POE::Wheel::Run->new(
Program => $cmd, Program => $cmd,
StdinFilter => POE::Filter::Line->new(), StdinFilter => POE::Filter::Line->new(),
StdoutFilter => POE::Filter::TCStatistics->new(), StdoutFilter => opentrafficshaper::POE::Filter::TCStatistics->new(),
StderrFilter => POE::Filter::Line->new(), StderrFilter => POE::Filter::Line->new(),
StdoutEvent => '_task_child_stdout', StdoutEvent => '_task_child_stdout',
StderrEvent => '_task_child_stderr', StderrEvent => '_task_child_stderr',
...@@ -190,9 +209,9 @@ sub _session_tick ...@@ -190,9 +209,9 @@ sub _session_tick
$heap->{task_by_pid}->{$task->PID} = $task; $heap->{task_by_pid}->{$task->PID} = $task;
# Signal events include the process ID. # Signal events include the process ID.
$heap->{task_data}->{$task->ID} = { $heap->{task_data}->{$task->ID} = {
'timestamp' => $now, 'Timestamp' => $now,
'interface' => $interface, 'Interface' => $interfaceID,
'current_stat' => { } 'CurrentStat' => { }
}; };
# Build commandline string # Build commandline string
...@@ -200,20 +219,32 @@ sub _session_tick ...@@ -200,20 +219,32 @@ sub _session_tick
$logger->log(LOG_DEBUG,"[TCSTATS] TASK/%s: Starting '%s' as %s with PID %s",$task->ID,$cmdStr,$task->ID,$task->PID); $logger->log(LOG_DEBUG,"[TCSTATS] TASK/%s: Starting '%s' as %s with PID %s",$task->ID,$cmdStr,$task->ID,$task->PID);
# Set last time we were run to now # Set last time we were run to now
$lastStats->{$interface} = $now; $globals->{'LastStats'}->{$interfaceID} = $now;
# NK: Space the stats out, this will cause TICK_PERIOD to elapse before we do another interface # Grab next one for below calcs...
$interfaceCount++; $interfaceID = shift(@interfaces);
last;
} }
# If we didn't fire up any stats, re-tick # Set default tick period
if (!$interfaceCount) { my $tickPeriod = opentrafficshaper::plugins::statistics::STATISTICS_PERIOD;
$kernel->delay('_tick' => TICK_PERIOD); # Calculate optimal wait time
if (defined($interfaceID)) {
$tickPeriod = opentrafficshaper::plugins::statistics::STATISTICS_PERIOD - ($now - $globals->{'LastStats'}->{$interfaceID});
}
# Make sure wait time is not too long
if ($tickPeriod > opentrafficshaper::plugins::statistics::STATISTICS_PERIOD) {
$tickPeriod = opentrafficshaper::plugins::statistics::STATISTICS_PERIOD;
}
# Make sure wait time is not too short
if ($tickPeriod < opentrafficshaper::plugins::statistics::STATISTICS_PERIOD) {
$tickPeriod = opentrafficshaper::plugins::statistics::STATISTICS_PERIOD;
} }
$kernel->delay('_tick' => $tickPeriod);
}; };
# Child writes to STDOUT # Child writes to STDOUT
sub _task_child_stdout sub _task_child_stdout
{ {
...@@ -225,8 +256,8 @@ sub _task_child_stdout ...@@ -225,8 +256,8 @@ sub _task_child_stdout
# Grab task data # Grab task data
my $taskData = $heap->{'task_data'}->{$task_id}; my $taskData = $heap->{'task_data'}->{$task_id};
my $interface = $taskData->{'interface'}; my $interface = $taskData->{'Interface'};
my $timestamp = $taskData->{'timestamp'}; my $timestamp = $taskData->{'Timestamp'};
# Stats ID to update # Stats ID to update
my $sid; my $sid;
...@@ -234,18 +265,18 @@ sub _task_child_stdout ...@@ -234,18 +265,18 @@ sub _task_child_stdout
my $direction = opentrafficshaper::plugins::statistics::STATISTICS_DIR_TX; my $direction = opentrafficshaper::plugins::statistics::STATISTICS_DIR_TX;
# Is this a system class? # Is this a system class?
my $classChildDec = hex($stat->{'_class_child'}); my $classChildDec = hex($stat->{'TCClassChild'});
# Check if this is a limit class... # Check if this is a limit class...
if (opentrafficshaper::plugins::tc::isPoolTcClass($interface,$stat->{'_class_parent'},$stat->{'_class_child'})) { if (opentrafficshaper::plugins::tc::isPoolTcClass($interface,$stat->{'TCClassParent'},$stat->{'TCClassChild'})) {
if (defined(my $pid = opentrafficshaper::plugins::tc::getPIDFromTcClass($interface,$stat->{'_class_parent'}, if (defined(my $pid = opentrafficshaper::plugins::tc::getPIDFromTcClass($interface,$stat->{'TCClassParent'},
$stat->{'_class_child'})) $stat->{'TCClassChild'}))
) { ) {
$sid = opentrafficshaper::plugins::statistics::setSIDFromPID($pid); $sid = opentrafficshaper::plugins::statistics::setSIDFromPID($pid);
$direction = opentrafficshaper::plugins::statistics::getTrafficDirection($pid,$interface); $direction = opentrafficshaper::plugins::statistics::getTrafficDirection($pid,$interface);
} else { } else {
$logger->log(LOG_WARN,"[TCSTATS] Pool traffic class '%s:%s' NOT FOUND",$stat->{'_class_parent'}, $logger->log(LOG_WARN,"[TCSTATS] Pool traffic class '%s:%s' NOT FOUND",$stat->{'TCClassParent'},
$stat->{'_class_child'} $stat->{'TCClassChild'}
); );
} }
...@@ -259,12 +290,12 @@ sub _task_child_stdout ...@@ -259,12 +290,12 @@ sub _task_child_stdout
} else { } else {
# Save the class with the decimal number # Save the class with the decimal number
if (my $classID = opentrafficshaper::plugins::tc::getCIDFromTcClass($interface, if (my $classID = opentrafficshaper::plugins::tc::getCIDFromTcClass($interface,
opentrafficshaper::plugins::tc::TC_ROOT_CLASS,$stat->{'_class_child'}) opentrafficshaper::plugins::tc::TC_ROOT_CLASS,$stat->{'TCClassChild'})
) { ) {
$sid = opentrafficshaper::plugins::statistics::setSIDFromCID($interface,$classID); $sid = opentrafficshaper::plugins::statistics::setSIDFromCID($interface,$classID);
} else { } else {
$logger->log(LOG_WARN,"[TCSTATS] System traffic class '%s:%s' NOT FOUND",$stat->{'_class_parent'}, $logger->log(LOG_WARN,"[TCSTATS] System traffic class '%s:%s' NOT FOUND",$stat->{'TCClassParent'},
$stat->{'_class_child'} $stat->{'TCClassChild'}
); );
} }
} }
...@@ -273,14 +304,15 @@ sub _task_child_stdout ...@@ -273,14 +304,15 @@ sub _task_child_stdout
# Make sure we have the lid now # Make sure we have the lid now
if (defined($sid)) { if (defined($sid)) {
# Build our submission # Build our submission
$stat->{'timestamp'} = $timestamp; $stat->{'Timestamp'} = $timestamp;
$stat->{'direction'} = $direction; $stat->{'Direction'} = $direction;
$taskData->{'stats'}->{$sid} = $stat; $taskData->{'Stats'}->{$sid} = $stat;
} }
} }
# Child writes to STDERR # Child writes to STDERR
sub _task_child_stderr sub _task_child_stderr
{ {
...@@ -293,6 +325,7 @@ sub _task_child_stderr ...@@ -293,6 +325,7 @@ sub _task_child_stderr
} }
# Child closed its handles, it won't communicate with us, so remove it # Child closed its handles, it won't communicate with us, so remove it
sub _task_child_close sub _task_child_close
{ {
...@@ -309,7 +342,7 @@ sub _task_child_close ...@@ -309,7 +342,7 @@ sub _task_child_close
} }
# Push consolidated update through # Push consolidated update through
$kernel->post("statistics" => "update" => $taskData->{'stats'}); $kernel->post("statistics" => "update" => $taskData->{'Stats'});
$logger->log(LOG_DEBUG,"[TCSTATS] TASK/%s: Closed PID %s",$task_id,$task->PID); $logger->log(LOG_DEBUG,"[TCSTATS] TASK/%s: Closed PID %s",$task_id,$task->PID);
...@@ -317,12 +350,10 @@ sub _task_child_close ...@@ -317,12 +350,10 @@ sub _task_child_close
delete($heap->{task_by_pid}->{$task->PID}); delete($heap->{task_by_pid}->{$task->PID});
delete($heap->{task_by_wid}->{$task_id}); delete($heap->{task_by_wid}->{$task_id});
delete($heap->{task_data}->{$task_id}); delete($heap->{task_data}->{$task_id});
# Fire up next tick
$kernel->delay('_tick' => TICK_PERIOD);
} }
# Reap the dead child # Reap the dead child
sub _task_handle_SIGCHLD sub _task_handle_SIGCHLD
{ {
...@@ -342,6 +373,7 @@ sub _task_handle_SIGCHLD ...@@ -342,6 +373,7 @@ sub _task_handle_SIGCHLD
} }
# Handle SIGINT # Handle SIGINT
sub _task_handle_SIGINT sub _task_handle_SIGINT
{ {
...@@ -359,5 +391,6 @@ sub _task_handle_SIGINT ...@@ -359,5 +391,6 @@ sub _task_handle_SIGINT
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper webserver module: configmanager page # OpenTrafficShaper webserver module: configmanager page
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -32,24 +32,35 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -32,24 +32,35 @@ our (@ISA,@EXPORT,@EXPORT_OK);
use DateTime; use DateTime;
use HTML::Entities; use HTML::Entities;
use HTTP::Status qw( :constants ); use HTTP::Status qw(
:constants
);
use URI::Escape; use URI::Escape;
use opentrafficshaper::logger; use awitpt::util qw(
use opentrafficshaper::plugins; isNumber ISNUMBER_ALLOW_ZERO
use opentrafficshaper::utils qw(
parseFormContent parseFormContent
parseURIQuery parseURIQuery
prettyUndef prettyUndef
); );
use opentrafficshaper::logger;
use opentrafficshaper::plugins;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
getInterfaces getInterfaces
getInterfaceTrafficClasses getInterface
getInterfaceRate
getTrafficClass
getAllTrafficClasses
getInterfaceTrafficClass
getEffectiveInterfaceTrafficClass2
changeInterfaceTrafficClass
getTrafficClassName isInterfaceIDValid
isTrafficClassIDValid
); );
...@@ -82,45 +93,27 @@ sub default ...@@ -82,45 +93,27 @@ sub default
} }
# Admin configuration # Admin configuration
sub admin_config sub admin_config
{ {
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
# Items for our form...
my @formElements = qw(
FriendlyName
Identifier
ClassID
TrafficLimitTx TrafficLimitTxBurst
TrafficLimitRx TrafficLimitRxBurst
Notes
);
# Grab stuff we need # Grab stuff we need
my $interfaces = getInterfaces(); my @interfaces = getInterfaces();
# Errors to display above the form
my @errors;
# Build content # Build content
my $content = ""; my $content = "";
# Header # Form header
$content .=<<EOF; $content .=<<EOF;
<!-- Config Tabs --> <legend>Interface Rate Setup</legend>
<ul class="nav nav-tabs" id="configTabs">
<li class="active"><a href="#interfaces" data-toggle="tab">Interfaces</a></li>
<li><a href="#system" data-toggle="tab">System (TODO)</a></li>
</ul>
<!-- Tab panes -->
<div class="tab-content">
<div class="tab-pane active" id="interfaces">
EOF EOF
# Title of the form, by default its an add form
my $formType = "Add";
my $formNoEdit = "";
# Form data # Form data
my $formData; my $formData;
...@@ -129,11 +122,87 @@ EOF ...@@ -129,11 +122,87 @@ EOF
if ($request->method eq "POST") { if ($request->method eq "POST") {
# Parse form data # Parse form data
my $form = parseFormContent($request->content); my $form = parseFormContent($request->content);
use Data::Dumper; warn "CONTENTDATA: ".Dumper($request->content);
use Data::Dumper; warn "FORMDATA: ".Dumper($form); # Loop with rate changes
my $rateChanges = { };
foreach my $elementName (keys %{$form}) {
my $rateChange = $form->{$elementName};
# Skip over blanks
if ($rateChange->{'value'} =~ /^\s*$/) {
next;
}
# Split off the various components of the element name
my ($item,$interfaceID,$trafficClassID) = ($elementName =~ /^((?:CIR|Limit))\[([a-z0-9:.]+)\]\[([0-9]+)\]$/);
# Make sure everything is defined
if (!defined($item) || !defined($interfaceID) || !defined($trafficClassID)) {
push(@errors,"Invalid data received");
last;
}
# Check interface exists
if (!defined($interfaceID = isInterfaceIDValid($interfaceID))) {
push(@errors,"Invalid data received, interface ID is invalid");
last;
}
# Check class is valid
if (
!defined($trafficClassID = isNumber($trafficClassID,ISNUMBER_ALLOW_ZERO)) ||
($trafficClassID && !isTrafficClassIDValid($trafficClassID))
) {
push(@errors,"Invalid class ID received for interface [$interfaceID]");
last;
}
# Check value is valid
if (!defined($rateChange->{'value'} = isNumber($rateChange->{'value'}))) {
push(@errors,"Invalid value received for interface [$interfaceID], class [$trafficClassID]");
last;
}
$rateChanges->{$interfaceID}->{$trafficClassID}->{$item} = $rateChange->{'value'};
}
# FIXME - check speed does not exceed inteface speed
# Check if there are no errors
if (!@errors) {
# Loop with interfaces
foreach my $interfaceID (keys %{$rateChanges}) {
my $trafficClasses = $rateChanges->{$interfaceID};
# Loop with traffic classes
foreach my $trafficClassID (keys %{$trafficClasses}) {
my $trafficClass = $trafficClasses->{$trafficClassID};
# Set some additional items we need
$trafficClass->{'InterfaceID'} = $interfaceID;
$trafficClass->{'TrafficClassID'} = $trafficClassID;
# Push changes
changeInterfaceTrafficClass($trafficClass);
}
}
return (HTTP_TEMPORARY_REDIRECT,"/configmanager");
}
} }
# Header
$content .=<<EOF;
<!-- Config Tabs -->
<ul class="nav nav-tabs" id="configTabs">
<li class="active"><a href="#interfaces" data-toggle="tab">Interfaces</a></li>
</ul>
<!-- Tab panes -->
<div class="tab-content">
<div class="tab-pane active" id="interfaces">
EOF
# Spit out errors if we have any
if (@errors > 0) {
foreach my $error (@errors) {
$content .= '<div class="alert alert-danger">'.encode_entities($error).'</div>';
}
}
# Interfaces tab setup # Interfaces tab setup
$content .=<<EOF; $content .=<<EOF;
<br /> <br />
...@@ -141,14 +210,21 @@ EOF ...@@ -141,14 +210,21 @@ EOF
<ul class="nav nav-tabs" id="configInterfaceTabs"> <ul class="nav nav-tabs" id="configInterfaceTabs">
EOF EOF
my $firstPaneActive = " active"; my $firstPaneActive = " active";
foreach my $interface (@{$interfaces}) { foreach my $interfaceID (@interfaces) {
my $interface = getInterface($interfaceID);
my $encodedInterfaceID = encode_entities($interfaceID);
my $encodedInterfaceName = encode_entities($interface->{'Name'});
$content .=<<EOF; $content .=<<EOF;
<li class="$firstPaneActive"><a href="#interface$interface" data-toggle="tab">Interface: $interface</a></li> <li class="$firstPaneActive">
<a href="#interface$encodedInterfaceID" data-toggle="tab">
Interface: $encodedInterfaceName
</a>
</li>
EOF EOF
# No longer the first pane # No longer the first pane
$firstPaneActive = ""; $firstPaneActive = "";
$formData->{"MainTrafficLimitTx[$interface]"} = getInterfaceRate($interface);
} }
$content .=<<EOF; $content .=<<EOF;
</ul> </ul>
...@@ -158,17 +234,31 @@ EOF ...@@ -158,17 +234,31 @@ EOF
# Suck in list of interfaces # Suck in list of interfaces
$firstPaneActive = " active"; $firstPaneActive = " active";
foreach my $interface (@{$interfaces}) { foreach my $interfaceID (@interfaces) {
my $interface = getInterface($interfaceID);
my $encodedInterfaceID = encode_entities($interfaceID);
my $encodedInterfaceName = encode_entities($interface->{'Name'});
# Root class
my $interfaceTrafficClass = getInterfaceTrafficClass($interfaceID,0);
my $effectiveInterfaceTrafficClass = getEffectiveInterfaceTrafficClass2($interfaceTrafficClass->{'ID'});
my $encodedInterfaceLimit = encode_entities($effectiveInterfaceTrafficClass->{'Limit'});
# Interface tab # Interface tab
$content .=<<EOF; $content .=<<EOF;
<div class="tab-pane$firstPaneActive" id="interface$interface"> <div class="tab-pane$firstPaneActive" id="interface$encodedInterfaceID">
EOF EOF
# No longer the first pane # No longer the first pane
$firstPaneActive = ""; $firstPaneActive = "";
# Sanitize params if we need to # Sanitize params if we need to
foreach my $item (@formElements) { if (defined($formData->{"Limit[$encodedInterfaceID][0]"})) {
$formData->{$item} = defined($formData->{$item}) ? encode_entities($formData->{$item}) : ""; $formData->{"Limit[$encodedInterfaceID][0]"} =
encode_entities($formData->{"Limit[$encodedInterfaceID][0]"});
} else {
$formData->{"Limit[$encodedInterfaceID][0]"} = "";
} }
...@@ -183,52 +273,73 @@ EOF ...@@ -183,52 +273,73 @@ EOF
# #
$content .=<<EOF; $content .=<<EOF;
<br /> <br />
<legend>Main: $interface</legend> <legend>Main: $encodedInterfaceName</legend>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTx" class="col-md-1 control-label">CIR</label> <label for="Limit" class="col-md-1 control-label">Speed</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="MainTrafficLimitTx[$interface]" type="text" placeholder="CIR" class="form-control" value="$formData->{"MainTrafficLimitTx[$interface]"}" /> <input name="Limit[$encodedInterfaceID][0]" type="text"
placeholder="$encodedInterfaceLimit" class="form-control"
value="$formData->{"Limit[$encodedInterfaceID][0]"}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
<label for="TrafficLimitTxBurst" class="col-md-1 control-label">Limit</label>
<div class="col-md-3">
<div class="input-group">
<input name="MainTrafficLimitTxBurst[$interface]" type="text" placeholder="Limit" class="form-control" value="$formData->{'TrafficLimitTxBurst'}" />
<span class="input-group-addon">Kbps<span>
</div>
</div>
</div> </div>
</div> </div>
EOF EOF
my $classes = getInterfaceTrafficClasses($interface); # Grab classes and loop
foreach my $class (sort { $a <=> $b } keys %{$classes}) { my @trafficClasses = getAllTrafficClasses();
my $className = getTrafficClassName($class); foreach my $trafficClassID (sort { $a <=> $b } @trafficClasses) {
my $classNameStr = encode_entities($className); my $trafficClass = getTrafficClass($trafficClassID);
my $encodedTrafficClassID = encode_entities($trafficClassID);
my $encodedTrafficClassName = encode_entities($trafficClass->{'Name'});
$interfaceTrafficClass = getInterfaceTrafficClass($interfaceID,$trafficClassID);
$effectiveInterfaceTrafficClass = getEffectiveInterfaceTrafficClass2($interfaceTrafficClass->{'ID'});
my $encodedInterfaceTrafficClassCIR = encode_entities($effectiveInterfaceTrafficClass->{'CIR'});
my $encodedInterfaceTrafficClassLimit = encode_entities($effectiveInterfaceTrafficClass->{'Limit'});
# Sanitize params if we need to
if (defined($formData->{"CIR[$encodedInterfaceID][$encodedTrafficClassID]"})) {
$formData->{"CIR[$encodedInterfaceID][$encodedTrafficClassID]"} =
encode_entities($formData->{"CIR[$encodedInterfaceID][$encodedTrafficClassID]"});
} else {
$formData->{"CIR[$encodedInterfaceID][$encodedTrafficClassID]"} = "";
}
if (defined($formData->{"Limit[$encodedInterfaceID][$encodedTrafficClassID]"})) {
$formData->{"Limit[$encodedInterfaceID][$encodedTrafficClassID]"} =
encode_entities($formData->{"Limit[$encodedInterfaceID][$encodedTrafficClassID]"});
} else {
$formData->{"Limit[$encodedInterfaceID][$encodedTrafficClassID]"} = "";
}
# #
# Page content # Page content
# #
$content .=<<EOF; $content .=<<EOF;
<legend>Class: $interface - $classNameStr</legend> <legend>Class: $encodedInterfaceName - $encodedTrafficClassName</legend>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTx" class="col-md-1 control-label">CIR</label> <label for="TxCIR" class="col-md-1 control-label">CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="ClassTrafficLimitTx[$interface] x y [$class]" type="text" placeholder="CIR" class="form-control" value="$formData->{'TrafficLimitTx'}" /> <input name="CIR[$encodedInterfaceID][$encodedTrafficClassID]" type="text"
placeholder="$encodedInterfaceTrafficClassCIR" class="form-control"
value="$formData->{"CIR[$encodedInterfaceID][$encodedTrafficClassID]"}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
<label for="TrafficLimitTxBurst" class="col-md-1 control-label">Limit</label> <label for="TxLimit" class="col-md-1 control-label">Limit</label>
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="ClassTrafficLimitTxBurst[$interface][$class]" type="text" placeholder="Limit" class="form-control" value="$formData->{'TrafficLimitTxBurst'}" /> <input name="Limit[$encodedInterfaceID][$encodedTrafficClassID]" type="text"
placeholder="$encodedInterfaceTrafficClassLimit" class="form-control"
value="$formData->{"Limit[$encodedInterfaceID][$encodedTrafficClassID]"}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -256,9 +367,6 @@ EOF ...@@ -256,9 +367,6 @@ EOF
EOF EOF
$content .=<<EOF; $content .=<<EOF;
<div class="tab-pane" id="system">
Work In Progress
</div>
</div> </div>
EOF EOF
......
# OpenTrafficShaper webserver module: index page # OpenTrafficShaper webserver module: index page
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -29,6 +29,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -29,6 +29,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
@EXPORT_OK = qw( @EXPORT_OK = qw(
); );
use HTTP::Status qw( :constants );
use opentrafficshaper::plugins; use opentrafficshaper::plugins;
...@@ -39,100 +40,18 @@ sub _catchall ...@@ -39,100 +40,18 @@ sub _catchall
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
# Build content my ($res,$content,$opts);
my $content = "";
if (!isPluginLoaded('statistics')) { if (!isPluginLoaded('statistics')) {
$content .= "No Statistics Plugin"; $content .= "No Statistics Plugin";
$res = HTTP_OK;
goto END; goto END;
} }
my @leftGraphs; return (HTTP_TEMPORARY_REDIRECT,"statistics/dashboard");
my @rightGraphs;
for (my $i = 0; $i < 7; $i++) {
push(@leftGraphs,"Class $i");
}
for (my $i = 0; $i < 2; $i++) {
push(@rightGraphs,"Main $i");
}
# Loop while we have graphs to output
while (@leftGraphs || @rightGraphs) {
# Layout Begin
$content .= <<EOF;
<div class="row">
EOF
# LHS - Begin
$content .= <<EOF;
<div class="col-xs-8">
EOF
# Loop with 2 sets of normal graphs per row
for (my $row = 0; $row < 2; $row++) {
# LHS - Begin Row
$content .= <<EOF;
<div class="row">
<div class="col-xs-6">
EOF
# Graph 1
if (defined(my $graph = shift(@leftGraphs))) {
$content .= <<EOF;
<h4 style="color:#8f8f8f;">Latest Data For: $graph</h4>
<div id="flotCanvas" class="flotCanvas" style="width: 520px; height: 150px; border: 1px dashed black">
</div>
EOF
}
# LHS - Spacer
$content .= <<EOF;
</div>
<div class="col-xs-6">
EOF
# Graph 2
if (defined(my $graph = shift(@leftGraphs))) {
$content .= <<EOF;
<h4 style="color:#8f8f8f;">Latest Data For: $graph</h4>
<div id="flotCanvas" class="flotCanvas" style="width: 520px; height: 150px; border: 1px dashed black">
</div>
EOF
}
# LHS - End Row
$content .= <<EOF;
</div>
</div>
EOF
}
# LHS - End
$content .= <<EOF;
</div>
EOF
# RHS - Begin Row
$content .= <<EOF;
<div class="col-xs-4">
EOF
# Graph
if (defined(my $graph = shift(@rightGraphs))) {
$content .= <<EOF;
<h4 style="color:#8f8f8f;">Latest Data For: $graph</h4>
<div id="flotCanvas" class="flotCanvas" style="width: 520px; height: 340px; border: 1px dashed black"></div>
EOF
}
# RHS - End Row
$content .= <<EOF;
</div>
EOF
# Layout End
$content .= <<EOF;
</div>
EOF
}
END:
return (200,$content);
} }
1; 1;
# vim: ts=4 # vim: ts=4
# OpenTrafficShaper webserver module: limits page # OpenTrafficShaper webserver module: limits page
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -32,28 +32,34 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -32,28 +32,34 @@ our (@ISA,@EXPORT,@EXPORT_OK);
use DateTime; use DateTime;
use HTML::Entities; use HTML::Entities;
use HTTP::Status qw( :constants ); use HTTP::Status qw(
use URI::Escape qw( uri_escape ); :constants
);
use NetAddr::IP;
use URI::Escape qw(
uri_escape
);
use URI::QueryParam; use URI::QueryParam;
use Storable qw( dclone ); use Storable qw(
dclone
);
use opentrafficshaper::constants; use awitpt::util qw(
use opentrafficshaper::logger;
use opentrafficshaper::plugins;
use opentrafficshaper::utils qw(
parseURIQuery parseURIQuery
parseFormContent parseFormContent
isUsername isUsername ISUSERNAME_ALLOW_ATSIGN
isIP isNumber ISNUMBER_ALLOW_ZERO
isNumber
prettyUndef prettyUndef
); );
use opentrafficshaper::constants;
use opentrafficshaper::logger;
use opentrafficshaper::plugins;
use opentrafficshaper::plugins::configmanager qw( use opentrafficshaper::plugins::configmanager qw(
getPools getPools
getPool getPool
getPoolByName getPoolByName
getPoolShaperState getPoolShaperState
isPoolOverridden
isPoolReady isPoolReady
getPoolMembers getPoolMembers
...@@ -62,19 +68,23 @@ use opentrafficshaper::plugins::configmanager qw( ...@@ -62,19 +68,23 @@ use opentrafficshaper::plugins::configmanager qw(
getPoolMemberShaperState getPoolMemberShaperState
isPoolMemberReady isPoolMemberReady
getOverrides getPoolOverrides
getOverride getPoolOverride
getInterfaceGroup
getInterfaceGroups getInterfaceGroups
isInterfaceGroupIDValid isInterfaceGroupIDValid
getMatchPriorities getMatchPriorities
isMatchPriorityIDValid isMatchPriorityIDValid
getTrafficClasses getTrafficClassName getTrafficClass
getTrafficClasses
isTrafficClassIDValid isTrafficClassIDValid
); );
use opentrafficshaper::util qw(
isIPv46 isIPv46CIDR
);
# Sidebar menu options for this module # Sidebar menu options for this module
...@@ -101,11 +111,11 @@ my $menu = [ ...@@ -101,11 +111,11 @@ my $menu = [
'items' => [ 'items' => [
{ {
'name' => 'List Overrides', 'name' => 'List Overrides',
'link' => 'override-list' 'link' => 'pool-override-list'
}, },
{ {
'name' => 'Add Override', 'name' => 'Add Override',
'link' => 'override-add' 'link' => 'pool-override-add'
} }
] ]
}, },
...@@ -146,6 +156,7 @@ sub pool_list ...@@ -146,6 +156,7 @@ sub pool_list
<th>Friendly Name</th> <th>Friendly Name</th>
<th>Name</th> <th>Name</th>
<th>Expires</th> <th>Expires</th>
<th>Members</th>
<th></th> <th></th>
<th>Class</th> <th>Class</th>
<th>CIR (Kbps)</th> <th>CIR (Kbps)</th>
...@@ -173,21 +184,23 @@ EOF ...@@ -173,21 +184,23 @@ EOF
# Get a nice last update string # Get a nice last update string
my $lastUpdate = DateTime->from_epoch( epoch => $pool->{'LastUpdate'} )->iso8601(); my $lastUpdate = DateTime->from_epoch( epoch => $pool->{'LastUpdate'} )->iso8601();
my $cirStr = sprintf('%s/%s',prettyUndef($pool->{'TrafficLimitTx'}),prettyUndef($pool->{'TrafficLimitRx'})); my $poolCIRStr = encode_entities(sprintf('%s/%s',prettyUndef($pool->{'TxCIR'}),prettyUndef($pool->{'RxCIR'})));
my $limitStr = sprintf('%s/%s',prettyUndef($pool->{'TrafficLimitTxBurst'}),prettyUndef($pool->{'TrafficLimitRxBurst'})); my $poolLimitStr = encode_entities(sprintf('%s/%s',prettyUndef($pool->{'TxLimit'}),prettyUndef($pool->{'RxLimit'})));
my $poolFriendlyName = (defined($pool->{'FriendlyName'}) && $pool->{'FriendlyName'} ne "") ? $pool->{'FriendlyName'} :
my $pidEscaped = uri_escape($pool->{'ID'}); $pool->{'Name'};
my $poolFriendlyNameEncoded = encode_entities($poolFriendlyName);
my $poolNameEncoded = encode_entities($pool->{'Name'});
my $friendlyName = (defined($pool->{'FriendlyName'}) && $pool->{'FriendlyName'} ne "") ? $pool->{'FriendlyName'} : my $poolExpiresStr = encode_entities(
$pool->{'Name'}; ($pool->{'Expires'} > 0) ? DateTime->from_epoch( epoch => $pool->{'Expires'} )->iso8601() : '-never-'
my $friendlyNameEncoded = encode_entities($friendlyName); );
my $nameEncoded = encode_entities($pool->{'Name'}); my $poolMemberCount = getPoolMembers($pool->{'ID'});
my $expiresStr = ($pool->{'Expires'} > 0) ? DateTime->from_epoch( epoch => $pool->{'Expires'} )->iso8601() : '-never-';
my $classStr = getTrafficClassName($pool->{'ClassID'}); my $trafficClass = getTrafficClass($pool->{'TrafficClassID'});
my $trafficClassNameEncoded = encode_entities($trafficClass->{'Name'});
# Display relevant icons depending on pool status # Display relevant icons depending on pool status
my $icons = ""; my $icons = "";
...@@ -203,26 +216,31 @@ EOF ...@@ -203,26 +216,31 @@ EOF
# if ($pool->{'Status'} eq 'conflict') { # if ($pool->{'Status'} eq 'conflict') {
# $icons .= '<span class="glyphicon glyphicon-random" />'; # $icons .= '<span class="glyphicon glyphicon-random" />';
# } # }
# if ($pool->{'Status'} eq 'conflict') { if (isPoolOverridden($pool->{'ID'})) {
# $icons .= '<span class="glyphicon glyphicon-edit" />'; $icons .= '<span class="glyphicon glyphicon-edit" />';
# } }
my $urlStatsPool = sprintf('/statistics/by-pool?pool=%s',uri_escape("$pool->{'InterfaceGroupID'}:$pool->{'Name'}"));
my $urlPoolEdit = sprintf('/limits/pool-edit?pid=%s',uri_escape($pool->{'ID'}));
my $urlPoolMemberList = sprintf('/limits/poolmember-list?pid=%s',uri_escape($pool->{'ID'}));
my $urlPoolRemove = sprintf('/limits/pool-remove?pid=%s',uri_escape($pool->{'ID'}));
$content .= <<EOF; $content .= <<EOF;
<tr> <tr>
<td>$icons</td> <td>$icons</td>
<td>$friendlyNameEncoded</td> <td>$poolFriendlyNameEncoded</td>
<td>$nameEncoded</td> <td>$poolNameEncoded</td>
<td>$expiresStr</td> <td>$poolExpiresStr</td>
<td class="align-right">$poolMemberCount</td>
<td><span class="glyphicon glyphicon-arrow-right" /></td> <td><span class="glyphicon glyphicon-arrow-right" /></td>
<td>$classStr</td> <td class="align-center">$trafficClassNameEncoded</td>
<td>$cirStr</td> <td class="align-center">$poolCIRStr</td>
<td>$limitStr</td> <td class="align-center">$poolLimitStr</td>
<td> <td>
<a href="/statistics/by-pool?pid=$pidEscaped"><span class="glyphicon glyphicon-stats"></span></a> <a href="$urlStatsPool"><span class="glyphicon glyphicon-stats"></span></a>
<a href="/limits/pool-edit?pid=$pidEscaped"><span class="glyphicon glyphicon-wrench"></span></a> <a href="$urlPoolEdit"><span class="glyphicon glyphicon-wrench"></span></a>
<a href="/limits/poolmember-list?pid=$pidEscaped"><span class="glyphicon glyphicon-link"></span></a> <a href="$urlPoolMemberList"><span class="glyphicon glyphicon-link"></span></a>
<a href="/limits/pool-remove?pid=$pidEscaped"><span class="glyphicon glyphicon-remove"></span></a> <a href="$urlPoolRemove"><span class="glyphicon glyphicon-remove"></span></a>
</td> </td>
</tr> </tr>
EOF EOF
...@@ -232,7 +250,7 @@ EOF ...@@ -232,7 +250,7 @@ EOF
if (!@pools) { if (!@pools) {
$content .=<<EOF; $content .=<<EOF;
<tr class="info"> <tr class="info">
<td colspan="8"><p class="text-center">No Results</p></td> <td colspan="10"><p class="text-center">No Results</p></td>
</tr> </tr>
EOF EOF
} }
...@@ -241,10 +259,10 @@ EOF ...@@ -241,10 +259,10 @@ EOF
$content .=<<EOF; $content .=<<EOF;
</tbody> </tbody>
</table> </table>
<span class="glyphicon glyphicon-time" /> - Processing, <span class="glyphicon glyphicon-time" /> - Processing <br/>
<span class="glyphicon glyphicon-edit" /> - Override, <span class="glyphicon glyphicon-edit" /> - Override <br/>
<span class="glyphicon glyphicon-import" /> - Being Added, <span class="glyphicon glyphicon-import" /> - Being Added <br/>
<span class="glyphicon glyphicon-trash" /> - Being Removed, <span class="glyphicon glyphicon-trash" /> - Being Removed <br/>
<span class="glyphicon glyphicon-random" /> - Conflicts <span class="glyphicon glyphicon-random" /> - Conflicts
EOF EOF
...@@ -252,6 +270,7 @@ EOF ...@@ -252,6 +270,7 @@ EOF
} }
# Pool add/edit action # Pool add/edit action
sub pool_addedit sub pool_addedit
{ {
...@@ -269,9 +288,9 @@ sub pool_addedit ...@@ -269,9 +288,9 @@ sub pool_addedit
FriendlyName FriendlyName
Name Name
InterfaceGroupID InterfaceGroupID
ClassID TrafficClassID
TrafficLimitTx TrafficLimitTxBurst TxCIR TxLimit
TrafficLimitRx TrafficLimitRxBurst RxCIR RxLimit
Expires inputExpires.modifier Expires inputExpires.modifier
Notes Notes
); );
...@@ -300,7 +319,7 @@ sub pool_addedit ...@@ -300,7 +319,7 @@ sub pool_addedit
if (defined($queryParams->{'pid'})) { if (defined($queryParams->{'pid'})) {
# Check if we can grab the pool # Check if we can grab the pool
if (!defined($pool = getPool($queryParams->{'pid'}->{'value'}))) { if (!defined($pool = getPool($queryParams->{'pid'}->{'value'}))) {
return (HTTP_TEMPORARY_REDIRECT,"limits"); return (HTTP_TEMPORARY_REDIRECT,"/limits");
} }
} }
...@@ -312,7 +331,7 @@ sub pool_addedit ...@@ -312,7 +331,7 @@ sub pool_addedit
# If user pressed cancel, redirect # If user pressed cancel, redirect
if (defined($form->{'cancel'})) { if (defined($form->{'cancel'})) {
# Redirects to default page # Redirects to default page
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
# Transform form into form data # Transform form into form data
...@@ -324,7 +343,7 @@ sub pool_addedit ...@@ -324,7 +343,7 @@ sub pool_addedit
if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") { if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") {
# Check pool exists # Check pool exists
if (!defined($pool)) { if (!defined($pool)) {
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
$formData->{'ID'} = $pool->{'ID'}; $formData->{'ID'} = $pool->{'ID'};
...@@ -346,7 +365,7 @@ sub pool_addedit ...@@ -346,7 +365,7 @@ sub pool_addedit
# Woops ... no query string? # Woops ... no query string?
} elsif (keys %{$queryParams} > 0) { } elsif (keys %{$queryParams} > 0) {
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
} }
...@@ -357,7 +376,7 @@ sub pool_addedit ...@@ -357,7 +376,7 @@ sub pool_addedit
# Check POST data # Check POST data
my $name; my $name;
if (!defined($name = isUsername($formData->{'Name'}))) { if (!defined($name = isUsername($formData->{'Name'},ISUSERNAME_ALLOW_ATSIGN))) {
push(@errors,"Name is not valid"); push(@errors,"Name is not valid");
} }
my $interfaceGroupID; my $interfaceGroupID;
...@@ -367,18 +386,18 @@ sub pool_addedit ...@@ -367,18 +386,18 @@ sub pool_addedit
if ($formType ne "Edit" && getPoolByName($interfaceGroupID,$name)) { if ($formType ne "Edit" && getPoolByName($interfaceGroupID,$name)) {
push(@errors,"A pool with the same name already exists"); push(@errors,"A pool with the same name already exists");
} }
my $classID; my $trafficClassID;
if (!defined($classID = isTrafficClassIDValid($formData->{'ClassID'}))) { if (!defined($trafficClassID = isTrafficClassIDValid($formData->{'TrafficClassID'}))) {
push(@errors,"Traffic class is not valid"); push(@errors,"Traffic class is not valid");
} }
my $trafficLimitTx = isNumber($formData->{'TrafficLimitTx'}); my $txCIR = isNumber($formData->{'TxCIR'});
my $trafficLimitTxBurst = isNumber($formData->{'TrafficLimitTxBurst'}); my $txLimit = isNumber($formData->{'TxLimit'});
if (!defined($trafficLimitTx) && !defined($trafficLimitTxBurst)) { if (!defined($txCIR) && !defined($txLimit)) {
push(@errors,"A valid download CIR and/or limit is required"); push(@errors,"A valid download CIR and/or limit is required");
} }
my $trafficLimitRx = isNumber($formData->{'TrafficLimitRx'}); my $rxCIR = isNumber($formData->{'RxCIR'});
my $trafficLimitRxBurst = isNumber($formData->{'TrafficLimitRxBurst'}); my $rxLimit = isNumber($formData->{'RxLimit'});
if (!defined($trafficLimitRx) && !defined($trafficLimitRxBurst)) { if (!defined($rxCIR) && !defined($rxLimit)) {
push(@errors,"A valid upload CIR and/or limit is required"); push(@errors,"A valid upload CIR and/or limit is required");
} }
...@@ -391,7 +410,7 @@ sub pool_addedit ...@@ -391,7 +410,7 @@ sub pool_addedit
my $expires = 0; my $expires = 0;
if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") { if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") {
if (!defined($expires = isNumber($formData->{'Expires'}))) { if (!defined($expires = isNumber($formData->{'Expires'},ISNUMBER_ALLOW_ZERO))) {
push(@errors,"Expires value is not valid"); push(@errors,"Expires value is not valid");
# Check the modifier # Check the modifier
} else { } else {
...@@ -429,11 +448,11 @@ sub pool_addedit ...@@ -429,11 +448,11 @@ sub pool_addedit
'FriendlyName' => $friendlyName, 'FriendlyName' => $friendlyName,
'Name' => $name, 'Name' => $name,
'InterfaceGroupID' => $interfaceGroupID, 'InterfaceGroupID' => $interfaceGroupID,
'ClassID' => $classID, 'TrafficClassID' => $trafficClassID,
'TrafficLimitTx' => $trafficLimitTx, 'TxCIR' => $txCIR,
'TrafficLimitTxBurst' => $trafficLimitTxBurst, 'TxLimit' => $txLimit,
'TrafficLimitRx' => $trafficLimitRx, 'RxCIR' => $rxCIR,
'TrafficLimitRxBurst' => $trafficLimitRxBurst, 'RxLimit' => $rxLimit,
'Expires' => $expires, 'Expires' => $expires,
'Notes' => $notes, 'Notes' => $notes,
}; };
...@@ -455,14 +474,14 @@ sub pool_addedit ...@@ -455,14 +474,14 @@ sub pool_addedit
$formType, $formType,
prettyUndef($name), prettyUndef($name),
prettyUndef($interfaceGroupID), prettyUndef($interfaceGroupID),
prettyUndef($classID), prettyUndef($trafficClassID),
prettyUndef($trafficLimitTx), prettyUndef($txCIR),
prettyUndef($trafficLimitRx), prettyUndef($rxCIR),
prettyUndef($trafficLimitTxBurst), prettyUndef($txLimit),
prettyUndef($trafficLimitRxBurst) prettyUndef($rxLimit)
); );
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
} }
...@@ -483,35 +502,43 @@ EOF ...@@ -483,35 +502,43 @@ EOF
# Spit out errors if we have any # Spit out errors if we have any
if (@errors > 0) { if (@errors > 0) {
foreach my $error (@errors) { foreach my $error (@errors) {
$content .= '<div class="alert alert-danger">'.$error.'</div>'; $content .= '<div class="alert alert-danger">'.encode_entities($error).'</div>';
} }
} }
# Generate interface group list # Generate interface group list
my $interfaceGroups = getInterfaceGroups(); my @interfaceGroups = sort(getInterfaceGroups());
my $interfaceGroupStr = ""; my $interfaceGroupStr = "";
foreach my $interfaceGroupID (sort keys %{$interfaceGroups}) { foreach my $interfaceGroupID (@interfaceGroups) {
my $interfaceGroup = getInterfaceGroup($interfaceGroupID);
my $interfaceGroupIDEncoded = encode_entities($interfaceGroupID);
my $interfaceGroupNameEncoded = encode_entities($interfaceGroup->{'Name'});
# Check if this item is selected # Check if this item is selected
my $selected = ""; my $selected = "";
if ($formData->{'InterfaceGroupID'} ne "" && $formData->{'InterfaceGroupID'} eq $interfaceGroupID) { if ($formData->{'InterfaceGroupID'} ne "" && $formData->{'InterfaceGroupID'} eq $interfaceGroupID) {
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$interfaceGroupStr .= '<option value="'.$interfaceGroupID.'" '.$selected.'>'. $interfaceGroupStr .= '<option value="'.$interfaceGroupIDEncoded.'" '.$selected.'>'.
$interfaceGroups->{$interfaceGroupID}->{'name'}.'</option>'; $interfaceGroupNameEncoded.'</option>';
} }
# Generate traffic class list # Generate traffic class list
my $trafficClasses = getTrafficClasses(); my @trafficClasses = sort(getTrafficClasses());
my $trafficClassStr = ""; my $trafficClassStr = "";
foreach my $classID (sort keys %{$trafficClasses}) { foreach my $trafficClassID (@trafficClasses) {
my $trafficClass = getTrafficClass($trafficClassID);
my $trafficClassIDEncoded = encode_entities($trafficClassID);
my $trafficClassNameEncoded = encode_entities($trafficClass->{'Name'});
# Process selections nicely # Process selections nicely
my $selected = ""; my $selected = "";
if ($formData->{'ClassID'} ne "" && $formData->{'ClassID'} eq $classID) { if ($formData->{'TrafficClassID'} ne "" && $formData->{'TrafficClassID'} eq $trafficClassID) {
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$trafficClassStr .= '<option value="'.$classID.'" '.$selected.'>'.$trafficClasses->{$classID}.'</option>'; $trafficClassStr .= '<option value="'.$trafficClassIDEncoded.'" '.$selected.'>'.$trafficClassNameEncoded.'</option>';
} }
# Generate expires modifiers list # Generate expires modifiers list
...@@ -527,8 +554,8 @@ EOF ...@@ -527,8 +554,8 @@ EOF
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.$expiresModifiers->{$expireModifier}. $expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.
'</option>'; encode_entities($expiresModifiers->{$expireModifier}).'</option>';
} }
# Blank expires if its 0 # Blank expires if its 0
...@@ -568,58 +595,58 @@ EOF ...@@ -568,58 +595,58 @@ EOF
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="ClassID" class="col-md-2 control-label">Traffic Class</label> <label for="TrafficClassID" class="col-md-2 control-label">Traffic Class</label>
<div class="row"> <div class="row">
<div class="col-md-2"> <div class="col-md-2">
<select name="ClassID" class="form-control"> <select name="TrafficClassID" class="form-control">
$trafficClassStr $trafficClassStr
</select> </select>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTx" class="col-md-2 control-label">Download CIR</label> <label for="TxCIR" class="col-md-2 control-label">Download CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTx" type="text" placeholder="Download CIR" class="form-control" <input name="TxCIR" type="text" placeholder="Download CIR" class="form-control"
value="$formData->{'TrafficLimitTx'}" /> value="$formData->{'TxCIR'}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTxBurst" class="col-md-2 control-label">Download Limit</label> <label for="TxLimit" class="col-md-2 control-label">Download Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTxBurst" type="text" placeholder="Download Limit" class="form-control" <input name="TxLimit" type="text" placeholder="Download Limit" class="form-control"
value="$formData->{'TrafficLimitTxBurst'}" /> value="$formData->{'TxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitRx" class="col-md-2 control-label">Upload CIR</label> <label for="RxCIR" class="col-md-2 control-label">Upload CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRx" type="text" placeholder="Upload CIR" class="form-control" <input name="RxCIR" type="text" placeholder="Upload CIR" class="form-control"
value="$formData->{'TrafficLimitRx'}" /> value="$formData->{'RxCIR'}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitRxBurst" class="col-md-2 control-label">Upload Limit</label> <label for="RxLimit" class="col-md-2 control-label">Upload Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRxBurst" type="text" placeholder="Upload Limit" class="form-control" <input name="RxLimit" type="text" placeholder="Upload Limit" class="form-control"
value="$formData->{'TrafficLimitRxBurst'}" /> value="$formData->{'RxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -659,6 +686,7 @@ EOF ...@@ -659,6 +686,7 @@ EOF
} }
# Pool remove action # Pool remove action
sub pool_remove sub pool_remove
{ {
...@@ -714,18 +742,16 @@ EOF ...@@ -714,18 +742,16 @@ EOF
# Post the removal # Post the removal
$kernel->post("configmanager" => "pool_remove" => $pool->{'ID'}); $kernel->post("configmanager" => "pool_remove" => $pool->{'ID'});
} }
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
# Make the pool ID safe for HTML
my $encodedPID = encode_entities($queryParams->{'pid'}->{'value'});
# Make the friendly name HTML safe # Make the friendly name HTML safe
my $encodedName = encode_entities($pool->{'Name'}); my $encodedPoolName = encode_entities($pool->{'Name'});
# Build our confirmation dialog # Build our confirmation dialog
$content .= <<EOF; $content .= <<EOF;
<div class="alert alert-danger"> <div class="alert alert-danger">
Are you very sure you wish to remove pool for "$encodedName"? Are you very sure you wish to remove pool for "$encodedPoolName"?
</div> </div>
<form role="form" method="post"> <form role="form" method="post">
<input type="submit" class="btn btn-primary" name="confirm" value="Yes" /> <input type="submit" class="btn btn-primary" name="confirm" value="Yes" />
...@@ -738,6 +764,7 @@ END: ...@@ -738,6 +764,7 @@ END:
} }
# Pool member list page/action # Pool member list page/action
sub poolmember_list sub poolmember_list
{ {
...@@ -778,14 +805,13 @@ EOF ...@@ -778,14 +805,13 @@ EOF
# Grab pools members # Grab pools members
my @poolMembers = getPoolMembers($pool->{'ID'}); my @poolMembers = getPoolMembers($pool->{'ID'});
# Make the pool ID safe for HTML
my $pidEscaped = encode_entities($pool->{'ID'});
my $poolFriendlyName = (defined($pool->{'FriendlyName'}) && $pool->{'FriendlyName'} ne "") ? $pool->{'FriendlyName'} : my $poolFriendlyName = (defined($pool->{'FriendlyName'}) && $pool->{'FriendlyName'} ne "") ? $pool->{'FriendlyName'} :
$pool->{'Name'}; $pool->{'Name'};
my $poolFriendlyNameEncoded = encode_entities($poolFriendlyName); my $poolFriendlyNameEncoded = encode_entities($poolFriendlyName);
my $poolNameEncoded = encode_entities($pool->{'Name'}); my $poolNameEncoded = encode_entities($pool->{'Name'});
my $urlPoolMemberAdd = sprintf('poolmember-add?pid=%s',uri_escape($pool->{'ID'}));
# Menu # Menu
$customMenu = [ $customMenu = [
{ {
...@@ -793,7 +819,7 @@ EOF ...@@ -793,7 +819,7 @@ EOF
'items' => [ 'items' => [
{ {
'name' => 'Add Pool Member', 'name' => 'Add Pool Member',
'link' => "poolmember-add?pid=$pidEscaped", 'link' => $urlPoolMemberAdd
}, },
], ],
}, },
...@@ -812,6 +838,7 @@ EOF ...@@ -812,6 +838,7 @@ EOF
<th>Friendly Name</th> <th>Friendly Name</th>
<th>Username</th> <th>Username</th>
<th>IP</th> <th>IP</th>
<th>NAT</th>
<th>Created</th> <th>Created</th>
<th>Updated</th> <th>Updated</th>
<th>Expires</th> <th>Expires</th>
...@@ -829,26 +856,30 @@ EOF ...@@ -829,26 +856,30 @@ EOF
} }
# Get a nice last update string # Get a nice last update string
my $pmidEscaped = uri_escape($poolMember->{'ID'}); my $poolMemberFriendlyName = (defined($poolMember->{'FriendlyName'}) && $poolMember->{'FriendlyName'} ne "") ?
my $friendlyName = (defined($poolMember->{'FriendlyName'}) && $poolMember->{'FriendlyName'} ne "") ?
$poolMember->{'FriendlyName'} : $poolMember->{'Username'}; $poolMember->{'FriendlyName'} : $poolMember->{'Username'};
my $friendlyNameEncoded = encode_entities($friendlyName); my $poolMemberFriendlyNameEncoded = encode_entities($poolMemberFriendlyName);
my $poolMemberUsernameEncoded = encode_entities($poolMember->{'Username'});
my $usernameEncoded = encode_entities($poolMember->{'Username'}); my $poolMemberIPEncoded = encode_entities($poolMember->{'IPAddress'});
my $poolMemberIPNATEncoded = encode_entities(
(defined($poolMember->{'IPNATAddress'}) && $poolMember->{'IPNATAddress'} ne "") ? $poolMember->{'IPNATAddress'} : '-none-'
);
my $natIcons = (defined($poolMember->{'IPNATInbound'}) && $poolMember->{'IPNATInbound'}) ? '<span class="glyphicon glyphicon-resize-vertical" />' : '';
my $ipEncoded = encode_entities($poolMember->{'IPAddress'}); my $poolMemberCreatedStr = encode_entities(($poolMember->{'Created'} > 0) ?
DateTime->from_epoch( epoch => $poolMember->{'Created'} )->iso8601() : '-never-');
my $poolMemberUpdatedStr = encode_entities(($poolMember->{'LastUpdate'} > 0) ?
DateTime->from_epoch( epoch => $poolMember->{'LastUpdate'} )->iso8601() : '-never-');
my $poolMemberExpiresStr = encode_entities(($poolMember->{'Expires'} > 0) ?
DateTime->from_epoch( epoch => $poolMember->{'Expires'} )->iso8601() : '-never-');
my $createdStr = ($poolMember->{'Created'} > 0) ? my $poolMemberShaperState = getPoolMemberShaperState($poolMember->{'ID'});
DateTime->from_epoch( epoch => $poolMember->{'Created'} )->iso8601() : '-never-';
my $updatedStr = ($poolMember->{'LastUpdate'} > 0) ?
DateTime->from_epoch( epoch => $poolMember->{'LastUpdate'} )->iso8601() : '-never-';
my $expiresStr = ($poolMember->{'Expires'} > 0) ?
DateTime->from_epoch( epoch => $poolMember->{'Expires'} )->iso8601() : '-never-';
# Display relevant icons depending on pool status # Display relevant icons depending on pool status
my $icons = ""; my $icons = "";
if (!(getPoolMemberShaperState($poolMember->{'ID'}) & SHAPER_LIVE)) { if (!($poolMemberShaperState & SHAPER_LIVE)) {
$icons .= '<span class="glyphicon glyphicon-time" />'; $icons .= '<span class="glyphicon glyphicon-time" />';
} }
if ($poolMember->{'Status'} == CFGM_NEW) { if ($poolMember->{'Status'} == CFGM_NEW) {
...@@ -857,25 +888,26 @@ EOF ...@@ -857,25 +888,26 @@ EOF
if ($poolMember->{'Status'} == CFGM_OFFLINE) { if ($poolMember->{'Status'} == CFGM_OFFLINE) {
$icons .= '<span class="glyphicon glyphicon-trash" />'; $icons .= '<span class="glyphicon glyphicon-trash" />';
} }
# if ($poolMember->{'Status'} eq 'conflict') { if ($poolMemberShaperState & SHAPER_CONFLICT) {
# $icons .= '<span class="glyphicon glyphicon-random" />'; $icons .= '<span class="glyphicon glyphicon-random" />';
# } }
# if ($pool->{'Status'} eq 'conflict') {
# $icons .= '<span class="glyphicon glyphicon-edit" />'; my $urlPoolMemberEdit = sprintf('/limits/poolmember-edit?pmid=%s',uri_escape($poolMember->{'ID'}));
# } my $urlPoolMemberRemove = sprintf('/limits/poolmember-remove?pmid=%s',uri_escape($poolMember->{'ID'}));
$content .= <<EOF; $content .= <<EOF;
<tr> <tr>
<td>$icons</td> <td>$icons</td>
<td>$friendlyNameEncoded</td> <td>$poolMemberFriendlyNameEncoded</td>
<td>$usernameEncoded</td> <td>$poolMemberUsernameEncoded</td>
<td>$ipEncoded</td> <td>$poolMemberIPEncoded</td>
<td>$createdStr</td> <td>$natIcons$poolMemberIPNATEncoded</td>
<td>$updatedStr</td> <td>$poolMemberCreatedStr</td>
<td>$expiresStr</td> <td>$poolMemberUpdatedStr</td>
<td>$poolMemberExpiresStr</td>
<td> <td>
<a href="/limits/poolmember-edit?pmid=$pmidEscaped"><span class="glyphicon glyphicon-wrench"></span></a> <a href="$urlPoolMemberEdit"><span class="glyphicon glyphicon-wrench"></span></a>
<a href="/limits/poolmember-remove?pmid=$pmidEscaped"><span class="glyphicon glyphicon-remove"></span></a> <a href="$urlPoolMemberRemove"><span class="glyphicon glyphicon-remove"></span></a>
</td> </td>
</tr> </tr>
EOF EOF
...@@ -894,10 +926,10 @@ EOF ...@@ -894,10 +926,10 @@ EOF
$content .=<<EOF; $content .=<<EOF;
</tbody> </tbody>
</table> </table>
<span class="glyphicon glyphicon-time" /> - Processing, <span class="glyphicon glyphicon-time" /> - Processing <br/>
<span class="glyphicon glyphicon-edit" /> - Override, <span class="glyphicon glyphicon-edit" /> - Override <br/>
<span class="glyphicon glyphicon-import" /> - Being Added, <span class="glyphicon glyphicon-import" /> - Being Added <br/>
<span class="glyphicon glyphicon-trash" /> - Being Removed, <span class="glyphicon glyphicon-trash" /> - Being Removed <br/>
<span class="glyphicon glyphicon-random" /> - Conflicts <span class="glyphicon glyphicon-random" /> - Conflicts
EOF EOF
...@@ -906,6 +938,7 @@ END: ...@@ -906,6 +938,7 @@ END:
} }
# Pool member add/edit action # Pool member add/edit action
sub poolmember_addedit sub poolmember_addedit
{ {
...@@ -921,7 +954,7 @@ sub poolmember_addedit ...@@ -921,7 +954,7 @@ sub poolmember_addedit
# Items for our form... # Items for our form...
my @formElements = qw( my @formElements = qw(
FriendlyName FriendlyName
Username IPAddress Username IPAddress IPNATAddress IPNATInbound
MatchPriorityID MatchPriorityID
Expires inputExpires.modifier Expires inputExpires.modifier
Notes Notes
...@@ -939,6 +972,7 @@ sub poolmember_addedit ...@@ -939,6 +972,7 @@ sub poolmember_addedit
# Title of the form, by default its an add form # Title of the form, by default its an add form
my $formType = "Add"; my $formType = "Add";
my $formNoEdit = ""; my $formNoEdit = "";
my $checkboxNoEdit = "";
# Form data # Form data
my $formData; my $formData;
# Pool # Pool
...@@ -953,7 +987,7 @@ sub poolmember_addedit ...@@ -953,7 +987,7 @@ sub poolmember_addedit
if (defined($queryParams->{'pmid'})) { if (defined($queryParams->{'pmid'})) {
# Check if we can grab the pool member # Check if we can grab the pool member
if (!defined($poolMember = getPoolMember($queryParams->{'pmid'}->{'value'}))) { if (!defined($poolMember = getPoolMember($queryParams->{'pmid'}->{'value'}))) {
return (HTTP_TEMPORARY_REDIRECT,"limits"); return (HTTP_TEMPORARY_REDIRECT,"/limits");
} }
$pool = getPool($poolMember->{'PoolID'}); $pool = getPool($poolMember->{'PoolID'});
...@@ -962,7 +996,7 @@ sub poolmember_addedit ...@@ -962,7 +996,7 @@ sub poolmember_addedit
} elsif (defined($queryParams->{'pid'})) { } elsif (defined($queryParams->{'pid'})) {
# Check if we can grab the pool # Check if we can grab the pool
if (!defined($pool = getPool($queryParams->{'pid'}->{'value'}))) { if (!defined($pool = getPool($queryParams->{'pid'}->{'value'}))) {
return (HTTP_TEMPORARY_REDIRECT,"limits"); return (HTTP_TEMPORARY_REDIRECT,"/limits");
} }
} }
...@@ -975,15 +1009,13 @@ sub poolmember_addedit ...@@ -975,15 +1009,13 @@ sub poolmember_addedit
if (defined($form->{'cancel'})) { if (defined($form->{'cancel'})) {
# If the pool member is defined, rededirect to pool member list # If the pool member is defined, rededirect to pool member list
if (defined($poolMember)) { if (defined($poolMember)) {
my $pidEscaped = uri_escape($poolMember->{'PoolID'}); return (HTTP_TEMPORARY_REDIRECT,sprintf('/limits/poolmember-list?pid=%s',$pool->{'ID'}));
return (HTTP_TEMPORARY_REDIRECT,"limits/poolmember-list?pid=$pidEscaped");
# Do same for pool # Do same for pool
} elsif (defined($pool)) { } elsif (defined($pool)) {
my $pidEscaped = uri_escape($pool->{'ID'}); return (HTTP_TEMPORARY_REDIRECT,sprintf('/limits/poolmember-list?pid=%s',$pool->{'ID'}));
return (HTTP_TEMPORARY_REDIRECT,"limits/poolmember-list?pid=$pidEscaped");
} }
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
# Transform form into form data # Transform form into form data
...@@ -995,16 +1027,17 @@ sub poolmember_addedit ...@@ -995,16 +1027,17 @@ sub poolmember_addedit
if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") { if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") {
# If there is no pool member on submit, redirect<F7> # If there is no pool member on submit, redirect<F7>
if (!defined($poolMember)) { if (!defined($poolMember)) {
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
$formData->{'ID'} = $poolMember->{'ID'}; $formData->{'ID'} = $poolMember->{'ID'};
$formType = "Edit"; $formType = "Edit";
$formNoEdit = "readonly"; $formNoEdit = "readonly";
$checkboxNoEdit = "disabled";
} }
# Maybe we were given an override key as a parameter? this would be an edit form # Maybe we were given a pool override key as a parameter? this would be an edit form
} elsif ($request->method eq "GET") { } elsif ($request->method eq "GET") {
# If we got a pool member, this is an edit # If we got a pool member, this is an edit
if (defined($poolMember)) { if (defined($poolMember)) {
...@@ -1016,10 +1049,11 @@ sub poolmember_addedit ...@@ -1016,10 +1049,11 @@ sub poolmember_addedit
# Lastly if we were given a key, this is actually an edit # Lastly if we were given a key, this is actually an edit
$formType = "Edit"; $formType = "Edit";
$formNoEdit = "readonly"; $formNoEdit = "readonly";
$checkboxNoEdit = "disabled";
# Woops ... no query string? # Woops ... no query string?
} elsif (!defined($pool)) { } elsif (!defined($pool)) {
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
} }
...@@ -1029,13 +1063,27 @@ sub poolmember_addedit ...@@ -1029,13 +1063,27 @@ sub poolmember_addedit
# Check POST data # Check POST data
my $username; my $username;
if (!defined($username = isUsername($formData->{'Username'}))) { if (!defined($username = isUsername($formData->{'Username'},ISUSERNAME_ALLOW_ATSIGN))) {
push(@errors,"Username is not valid"); push(@errors,"Username is not valid");
} }
my $ipAddress; my $ipAddress;
if (!defined($ipAddress = isIP($formData->{'IPAddress'}))) { if (!defined($ipAddress = isIPv46CIDR($formData->{'IPAddress'}))) {
push(@errors,"IP address is not valid"); push(@errors,"IP address is not valid");
} }
my $ipNATAddress;
if (defined($formData->{'IPNATAddress'}) && $formData->{'IPNATAddress'} ne "") {
if (!defined($ipNATAddress = isIPv46($formData->{'IPNATAddress'}))) {
push(@errors,"IP NAT address is not valid");
}
}
my $ipNATInbound;
if (defined($formData->{'IPNATInbound'}) && $formData->{'IPNATInbound'} ne "") {
if (defined($ipNATAddress)) {
$ipNATInbound = "yes";
} else {
push(@errors,"Cannot NAT inbound traffic if no NAT address is set");
}
}
my $matchPriorityID; my $matchPriorityID;
if (!defined($matchPriorityID = isMatchPriorityIDValid($formData->{'MatchPriorityID'}))) { if (!defined($matchPriorityID = isMatchPriorityIDValid($formData->{'MatchPriorityID'}))) {
push(@errors,"Match priority is not valid"); push(@errors,"Match priority is not valid");
...@@ -1053,7 +1101,7 @@ sub poolmember_addedit ...@@ -1053,7 +1101,7 @@ sub poolmember_addedit
my $expires = 0; my $expires = 0;
if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") { if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") {
if (!defined($expires = isNumber($formData->{'Expires'}))) { if (!defined($expires = isNumber($formData->{'Expires'},ISNUMBER_ALLOW_ZERO))) {
push(@errors,"Expires value is not valid"); push(@errors,"Expires value is not valid");
# Check the modifier # Check the modifier
} else { } else {
...@@ -1090,6 +1138,8 @@ sub poolmember_addedit ...@@ -1090,6 +1138,8 @@ sub poolmember_addedit
'FriendlyName' => $friendlyName, 'FriendlyName' => $friendlyName,
'Username' => $username, 'Username' => $username,
'IPAddress' => $ipAddress, 'IPAddress' => $ipAddress,
'IPNATAddress' => $ipNATAddress,
'IPNATInbound' => $ipNATInbound,
'GroupID' => 1, 'GroupID' => 1,
'MatchPriorityID' => $matchPriorityID, 'MatchPriorityID' => $matchPriorityID,
'Expires' => $expires, 'Expires' => $expires,
...@@ -1108,17 +1158,18 @@ sub poolmember_addedit ...@@ -1108,17 +1158,18 @@ sub poolmember_addedit
$kernel->post("configmanager" => $cEvent => $poolMemberData); $kernel->post("configmanager" => $cEvent => $poolMemberData);
$logger->log(LOG_INFO,'[WEBSERVER/POOLMEMBER] Account: %s, User: %s, IP: %s, Group: %s, MatchPriority: %s, Pool: %s', $logger->log(LOG_INFO,'[WEBSERVER/POOLMEMBER] Account: %s, User: %s, IP: %s, NAT: %s (inbound: %s), Group: %s, MatchPriority: %s, Pool: %s',
$formType, $formType,
prettyUndef($username), prettyUndef($username),
prettyUndef($ipAddress), prettyUndef($ipAddress),
prettyUndef($ipNATAddress),
prettyUndef($ipNATInbound),
prettyUndef(undef), prettyUndef(undef),
prettyUndef($matchPriorityID), prettyUndef($matchPriorityID),
prettyUndef($pool->{'ID'}), prettyUndef($pool->{'ID'}),
); );
my $pidEscaped = uri_escape($pool->{'ID'}); return (HTTP_TEMPORARY_REDIRECT,sprintf('/limits/poolmember-list?pid=%s',$pool->{'ID'}));
return (HTTP_TEMPORARY_REDIRECT,"limits/poolmember-list?pid=$pidEscaped");
} }
} }
...@@ -1155,7 +1206,7 @@ EOF ...@@ -1155,7 +1206,7 @@ EOF
# Spit out errors if we have any # Spit out errors if we have any
if (@errors > 0) { if (@errors > 0) {
foreach my $error (@errors) { foreach my $error (@errors) {
$content .= '<div class="alert alert-danger">'.$error.'</div>'; $content .= '<div class="alert alert-danger">'.encode_entities($error).'</div>';
} }
} }
...@@ -1190,10 +1241,14 @@ EOF ...@@ -1190,10 +1241,14 @@ EOF
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.$expiresModifiers->{$expireModifier}. $expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.
'</option>'; encode_entities($expiresModifiers->{$expireModifier}).'</option>';
} }
# If we have IPNATInbound set, we need to set it to checked
if (defined($formData->{'IPNATInbound'}) && $formData->{'IPNATInbound'} ne "") {
$formData->{'IPNATInbound'} = "checked";
}
# Blank expires if its 0 # Blank expires if its 0
if (defined($formData->{'Expires'}) && $formData->{'Expires'} eq "0") { if (defined($formData->{'Expires'}) && $formData->{'Expires'} eq "0") {
$formData->{'Expires'} = ""; $formData->{'Expires'} = "";
...@@ -1230,6 +1285,16 @@ EOF ...@@ -1230,6 +1285,16 @@ EOF
</div> </div>
</div> </div>
</div> </div>
<div class="form-group">
<label for="IPNATAddress" class="col-md-2 control-label">NAT Address</label>
<div class="row">
<div class="col-md-4 input-group">
<input name="IPNATAddress" type="text" placeholder="NAT Address" class="form-control"
value="$formData->{'IPNATAddress'}" $formNoEdit />
<input name="IPNATInbound" type="checkbox" $formData->{'IPNATInbound'} $checkboxNoEdit /> NAT Inbound
</div>
</div>
</div>
<div class="form-group"> <div class="form-group">
<label for="MatchPriorityID" class="col-md-2 control-label">Match Priority</label> <label for="MatchPriorityID" class="col-md-2 control-label">Match Priority</label>
<div class="row"> <div class="row">
...@@ -1274,6 +1339,7 @@ EOF ...@@ -1274,6 +1339,7 @@ EOF
} }
# Pool member remove action # Pool member remove action
sub poolmember_remove sub poolmember_remove
{ {
...@@ -1311,7 +1377,7 @@ EOF ...@@ -1311,7 +1377,7 @@ EOF
} }
# Make the pool ID safe for HTML # Make the pool ID safe for HTML
my $pidEscaped = encode_entities($poolMember->{'PoolID'}); my $urlPoolMemberAdd = sprintf('/limits/poolmember-add?pid=%s',encode_entities($poolMember->{'PoolID'}));
# Menu # Menu
$customMenu = [ $customMenu = [
...@@ -1320,7 +1386,7 @@ EOF ...@@ -1320,7 +1386,7 @@ EOF
'items' => [ 'items' => [
{ {
'name' => 'Add Pool Member', 'name' => 'Add Pool Member',
'link' => "poolmember-add?pid=$pidEscaped", 'link' => $urlPoolMemberAdd
}, },
], ],
}, },
...@@ -1337,19 +1403,19 @@ EOF ...@@ -1337,19 +1403,19 @@ EOF
# Post the removal # Post the removal
$kernel->post("configmanager" => "poolmember_remove" => $poolMember->{'ID'}); $kernel->post("configmanager" => "poolmember_remove" => $poolMember->{'ID'});
} }
return (HTTP_TEMPORARY_REDIRECT,"limits/poolmember-list?pid=$pidEscaped"); return (HTTP_TEMPORARY_REDIRECT,sprintf('/limits/poolmember-list?pid=%s',$poolMember->{'PoolID'}));
} }
# Make the friendly name HTML safe # Make the friendly name HTML safe
my $friendlyName = (defined($poolMember->{'FriendlyName'}) && $poolMember->{'FriendlyName'} ne "") ? my $poolMemberFriendlyName = (defined($poolMember->{'FriendlyName'}) && $poolMember->{'FriendlyName'} ne "") ?
$poolMember->{'FriendlyName'} : $poolMember->{'Username'}; $poolMember->{'FriendlyName'} : $poolMember->{'Username'};
my $friendlyNameEncoded = encode_entities($friendlyName); my $poolMemberFriendlyNameEncoded = encode_entities($poolMemberFriendlyName);
my $usernameEncoded = encode_entities($poolMember->{'Username'}); my $poolMemberUsernameEncoded = encode_entities($poolMember->{'Username'});
# Build our confirmation dialog # Build our confirmation dialog
$content .= <<EOF; $content .= <<EOF;
<div class="alert alert-danger"> <div class="alert alert-danger">
Are you very sure you wish to remove pool member "$friendlyNameEncoded" [$usernameEncoded]? Are you very sure you wish to remove pool member "$poolMemberFriendlyNameEncoded" [$poolMemberUsernameEncoded]?
</div> </div>
<form role="form" method="post"> <form role="form" method="post">
<input type="submit" class="btn btn-primary" name="confirm" value="Yes" /> <input type="submit" class="btn btn-primary" name="confirm" value="Yes" />
...@@ -1362,6 +1428,7 @@ END: ...@@ -1362,6 +1428,7 @@ END:
} }
# Add action # Add action
sub limit_add sub limit_add
{ {
...@@ -1377,12 +1444,12 @@ sub limit_add ...@@ -1377,12 +1444,12 @@ sub limit_add
# Items for our form... # Items for our form...
my @formElements = qw( my @formElements = qw(
FriendlyName FriendlyName
Username IPAddress Username IPAddress IPNATAddress IPNATInbound
InterfaceGroupID InterfaceGroupID
MatchPriorityID MatchPriorityID
ClassID TrafficClassID
TrafficLimitTx TrafficLimitTxBurst TxCIR TxLimit
TrafficLimitRx TrafficLimitRxBurst RxCIR RxLimit
Expires inputExpires.modifier Expires inputExpires.modifier
Notes Notes
); );
...@@ -1407,7 +1474,7 @@ sub limit_add ...@@ -1407,7 +1474,7 @@ sub limit_add
# If user pressed cancel, redirect # If user pressed cancel, redirect
if (defined($form->{'cancel'})) { if (defined($form->{'cancel'})) {
# Redirects to default page # Redirects to default page
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
# Transform form into form data # Transform form into form data
...@@ -1423,13 +1490,27 @@ sub limit_add ...@@ -1423,13 +1490,27 @@ sub limit_add
# Check POST data # Check POST data
my $username; my $username;
if (!defined($username = isUsername($formData->{'Username'}))) { if (!defined($username = isUsername($formData->{'Username'},ISUSERNAME_ALLOW_ATSIGN))) {
push(@errors,"Username is not valid"); push(@errors,"Username is not valid");
} }
my $ipAddress; my $ipAddress;
if (!defined($ipAddress = isIP($formData->{'IPAddress'}))) { if (!defined($ipAddress = isIPv46CIDR($formData->{'IPAddress'}))) {
push(@errors,"IP address is not valid"); push(@errors,"IP address is not valid");
} }
my $ipNATAddress;
if (defined($formData->{'IPNATAddress'}) && $formData->{'IPNATAddress'} ne "") {
if (!defined($ipNATAddress = isIPv46($formData->{'IPNATAddress'}))) {
push(@errors,"NAT address is not valid");
}
}
my $ipNATInbound;
if (defined($formData->{'IPNATInbound'}) && $formData->{'IPNATInbound'} ne "") {
if (defined($ipNATAddress)) {
$ipNATInbound = "yes";
} else {
push(@errors,"Cannot NAT inbound traffic if no NAT address is set");
}
}
my $interfaceGroupID; my $interfaceGroupID;
if (!defined($interfaceGroupID = isInterfaceGroupIDValid($formData->{'InterfaceGroupID'}))) { if (!defined($interfaceGroupID = isInterfaceGroupIDValid($formData->{'InterfaceGroupID'}))) {
push(@errors,"Interface group is not valid"); push(@errors,"Interface group is not valid");
...@@ -1438,24 +1519,24 @@ sub limit_add ...@@ -1438,24 +1519,24 @@ sub limit_add
if (!defined($matchPriorityID = isMatchPriorityIDValid($formData->{'MatchPriorityID'}))) { if (!defined($matchPriorityID = isMatchPriorityIDValid($formData->{'MatchPriorityID'}))) {
push(@errors,"Match priority is not valid"); push(@errors,"Match priority is not valid");
} }
my $classID; my $trafficClassID;
if (!defined($classID = isTrafficClassIDValid($formData->{'ClassID'}))) { if (!defined($trafficClassID = isTrafficClassIDValid($formData->{'TrafficClassID'}))) {
push(@errors,"Traffic class is not valid"); push(@errors,"Traffic class is not valid");
} }
my $trafficLimitTx = isNumber($formData->{'TrafficLimitTx'}); my $txCIR = isNumber($formData->{'TxCIR'});
my $trafficLimitTxBurst = isNumber($formData->{'TrafficLimitTxBurst'}); my $txLimit = isNumber($formData->{'TxLimit'});
if (!defined($trafficLimitTx) && !defined($trafficLimitTxBurst)) { if (!defined($txCIR) && !defined($txLimit)) {
push(@errors,"A valid download CIR and/or limit is required"); push(@errors,"A valid download CIR and/or limit is required");
} }
my $trafficLimitRx = isNumber($formData->{'TrafficLimitRx'}); my $rxCIR = isNumber($formData->{'RxCIR'});
my $trafficLimitRxBurst = isNumber($formData->{'TrafficLimitRxBurst'}); my $rxLimit = isNumber($formData->{'RxLimit'});
if (!defined($trafficLimitRx) && !defined($trafficLimitRxBurst)) { if (!defined($rxCIR) && !defined($rxLimit)) {
push(@errors,"A valid upload CIR and/or limit is required"); push(@errors,"A valid upload CIR and/or limit is required");
} }
my $expires = 0; my $expires = 0;
if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") { if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") {
if (!defined($expires = isNumber($formData->{'Expires'}))) { if (!defined($expires = isNumber($formData->{'Expires'},ISNUMBER_ALLOW_ZERO))) {
push(@errors,"Expires value is not valid"); push(@errors,"Expires value is not valid");
# Check the modifier # Check the modifier
} else { } else {
...@@ -1492,14 +1573,16 @@ sub limit_add ...@@ -1492,14 +1573,16 @@ sub limit_add
'FriendlyName' => $friendlyName, 'FriendlyName' => $friendlyName,
'Username' => $username, 'Username' => $username,
'IPAddress' => $ipAddress, 'IPAddress' => $ipAddress,
'IPNATAddress' => $ipNATAddress,
'IPNATInbound' => $ipNATInbound,
'GroupID' => 1, 'GroupID' => 1,
'InterfaceGroupID' => $interfaceGroupID, 'InterfaceGroupID' => $interfaceGroupID,
'MatchPriorityID' => $matchPriorityID, 'MatchPriorityID' => $matchPriorityID,
'ClassID' => $classID, 'TrafficClassID' => $trafficClassID,
'TrafficLimitTx' => $trafficLimitTx, 'TxCIR' => $txCIR,
'TrafficLimitTxBurst' => $trafficLimitTxBurst, 'TxLimit' => $txLimit,
'TrafficLimitRx' => $trafficLimitRx, 'RxCIR' => $rxCIR,
'TrafficLimitRxBurst' => $trafficLimitRxBurst, 'RxLimit' => $rxLimit,
'Expires' => $expires, 'Expires' => $expires,
'Notes' => $notes, 'Notes' => $notes,
}; };
...@@ -1510,21 +1593,23 @@ sub limit_add ...@@ -1510,21 +1593,23 @@ sub limit_add
$kernel->post("configmanager" => "limit_add" => $limit); $kernel->post("configmanager" => "limit_add" => $limit);
$logger->log(LOG_INFO,"[WEBSERVER/LIMITS] New User: %s, IP: %s, Group: %s, InterfaceGroup: %s, MatchPriority: %s, ". $logger->log(LOG_INFO,"[WEBSERVER/LIMITS] New User: %s, IP: %s, NAT: %s (inbound: %s), Group: %s, InterfaceGroup: %s, MatchPriority: %s, ".
"Class: %s, Limits: %s/%s, Burst: %s/%s", "Class: %s, Limits: %s/%s, Burst: %s/%s",
prettyUndef($username), prettyUndef($username),
prettyUndef($ipAddress), prettyUndef($ipAddress),
prettyUndef($ipNATAddress),
prettyUndef($ipNATInbound),
prettyUndef(undef), prettyUndef(undef),
prettyUndef($interfaceGroupID), prettyUndef($interfaceGroupID),
prettyUndef($matchPriorityID), prettyUndef($matchPriorityID),
prettyUndef($classID), prettyUndef($trafficClassID),
prettyUndef($trafficLimitTx), prettyUndef($txCIR),
prettyUndef($trafficLimitRx), prettyUndef($rxCIR),
prettyUndef($trafficLimitTxBurst), prettyUndef($txLimit),
prettyUndef($trafficLimitRxBurst) prettyUndef($rxLimit)
); );
return (HTTP_TEMPORARY_REDIRECT,'limits'); return (HTTP_TEMPORARY_REDIRECT,'/limits');
} }
} }
...@@ -1545,22 +1630,24 @@ EOF ...@@ -1545,22 +1630,24 @@ EOF
# Spit out errors if we have any # Spit out errors if we have any
if (@errors > 0) { if (@errors > 0) {
foreach my $error (@errors) { foreach my $error (@errors) {
$content .= '<div class="alert alert-danger">'.$error.'</div>'; $content .= '<div class="alert alert-danger">'.encode_entities($error).'</div>';
} }
} }
# Generate interface group list # Generate interface group list
my $interfaceGroups = getInterfaceGroups(); my @interfaceGroups = sort(getInterfaceGroups());
my $interfaceGroupStr = ""; my $interfaceGroupStr = "";
foreach my $interfaceGroupID (sort keys %{$interfaceGroups}) { foreach my $interfaceGroupID (@interfaceGroups) {
my $interfaceGroup = getInterfaceGroup($interfaceGroupID);
# Process selections nicely # Process selections nicely
my $selected = ""; my $selected = "";
if ($formData->{'InterfaceGroupID'} ne "" && $formData->{'InterfaceGroupID'} eq $interfaceGroupID) { if ($formData->{'InterfaceGroupID'} ne "" && $formData->{'InterfaceGroupID'} eq $interfaceGroupID) {
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$interfaceGroupStr .= '<option value="'.$interfaceGroupID.'" '.$selected.'>'. $interfaceGroupStr .= '<option value="'.encode_entities($interfaceGroupID).'" '.$selected.'>'.
$interfaceGroups->{$interfaceGroupID}->{'name'}.'</option>'; encode_entities($interfaceGroup->{'Name'}).'</option>';
} }
# Generate match priority list # Generate match priority list
...@@ -1577,21 +1664,24 @@ EOF ...@@ -1577,21 +1664,24 @@ EOF
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$matchPriorityStr .= '<option value="'.$matchPriorityID.'" '.$selected.'>'.$matchPriorities->{$matchPriorityID}. $matchPriorityStr .= '<option value="'.encode_entities($matchPriorityID).'" '.$selected.'>'.
'</option>'; encode_entities($matchPriorities->{$matchPriorityID}).'</option>';
} }
# Generate traffic class list # Generate traffic class list
my $trafficClasses = getTrafficClasses(); my @trafficClasses = sort(getTrafficClasses());
my $trafficClassStr = ""; my $trafficClassStr = "";
foreach my $classID (sort keys %{$trafficClasses}) { foreach my $trafficClassID (@trafficClasses) {
my $trafficClass = getTrafficClass($trafficClassID);
# Process selections nicely # Process selections nicely
my $selected = ""; my $selected = "";
if ($formData->{'ClassID'} ne "" && $formData->{'ClassID'} eq $classID) { if ($formData->{'TrafficClassID'} ne "" && $formData->{'TrafficClassID'} eq $trafficClassID) {
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$trafficClassStr .= '<option value="'.$classID.'" '.$selected.'>'.$trafficClasses->{$classID}.'</option>'; $trafficClassStr .= '<option value="'.encode_entities($trafficClassID).'" '.$selected.'>'.
encode_entities($trafficClass->{'Name'}).'</option>';
} }
# Generate expires modifiers list # Generate expires modifiers list
...@@ -1607,8 +1697,13 @@ EOF ...@@ -1607,8 +1697,13 @@ EOF
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.$expiresModifiers->{$expireModifier}. $expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.
'</option>'; encode_entities($expiresModifiers->{$expireModifier}).'</option>';
}
# If we have IPNATInbound set, we need to set it to checked
if (defined($formData->{'IPNATInbound'}) && $formData->{'IPNATInbound'} ne "") {
$formData->{'IPNATInbound'} = "checked";
} }
# Blank expires if its 0 # Blank expires if its 0
...@@ -1647,6 +1742,16 @@ EOF ...@@ -1647,6 +1742,16 @@ EOF
</div> </div>
</div> </div>
</div> </div>
<div class="form-group">
<label for="IPNATAddress" class="col-md-2 control-label">NAT Address</label>
<div class="row">
<div class="col-md-4 input-group">
<input name="IPNATAddress" type="text" placeholder="NAT Address" class="form-control"
value="$formData->{'IPNATAddress'}" />
<input name="IPNATInbound" type="checkbox" $formData->{'IPNATInbound'} /> NAT Inbound
</div>
</div>
</div>
<div class="form-group"> <div class="form-group">
<label for="InterfaceGroupID" class="col-md-2 control-label">Interface Group</label> <label for="InterfaceGroupID" class="col-md-2 control-label">Interface Group</label>
<div class="row"> <div class="row">
...@@ -1668,10 +1773,10 @@ EOF ...@@ -1668,10 +1773,10 @@ EOF
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="ClassID" class="col-md-2 control-label">Traffic Class</label> <label for="TrafficClassID" class="col-md-2 control-label">Traffic Class</label>
<div class="row"> <div class="row">
<div class="col-md-2"> <div class="col-md-2">
<select name="ClassID" class="form-control"> <select name="TrafficClassID" class="form-control">
$trafficClassStr $trafficClassStr
</select> </select>
</div> </div>
...@@ -1692,48 +1797,48 @@ EOF ...@@ -1692,48 +1797,48 @@ EOF
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTx" class="col-md-2 control-label">Download CIR</label> <label for="TxCIR" class="col-md-2 control-label">Download CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTx" type="text" placeholder="Download CIR" class="form-control" <input name="TxCIR" type="text" placeholder="Download CIR" class="form-control"
value="$formData->{'TrafficLimitTx'}" /> value="$formData->{'TxCIR'}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTxBurst" class="col-md-2 control-label">Download Limit</label> <label for="TxLimit" class="col-md-2 control-label">Download Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTxBurst" type="text" placeholder="Download Limit" class="form-control" <input name="TxLimit" type="text" placeholder="Download Limit" class="form-control"
value="$formData->{'TrafficLimitTxBurst'}" /> value="$formData->{'TxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitRx" class="col-md-2 control-label">Upload CIR</label> <label for="RxCIR" class="col-md-2 control-label">Upload CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRx" type="text" placeholder="Upload CIR" class="form-control" <input name="RxCIR" type="text" placeholder="Upload CIR" class="form-control"
value="$formData->{'TrafficLimitRx'}" /> value="$formData->{'RxCIR'}" />
<span class="input-group-addon">Kbps *<span> <span class="input-group-addon">Kbps *<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitRxBurst" class="col-md-2 control-label">Upload Limit</label> <label for="RxLimit" class="col-md-2 control-label">Upload Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRxBurst" type="text" placeholder="Upload Limit" class="form-control" <input name="RxLimit" type="text" placeholder="Upload Limit" class="form-control"
value="$formData->{'TrafficLimitRxBurst'}" /> value="$formData->{'RxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -1759,20 +1864,21 @@ EOF ...@@ -1759,20 +1864,21 @@ EOF
} }
# Override list
sub override_list # Pool override list
sub pool_override_list
{ {
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
my @overrides = getOverrides(); my @poolOverrides = getPoolOverrides();
# Build content # Build content
my $content = ""; my $content = "";
# Header # Header
$content .=<<EOF; $content .=<<EOF;
<legend>Override List</legend> <legend>Pool Override List</legend>
<table class="table"> <table class="table">
<thead> <thead>
<tr> <tr>
...@@ -1792,50 +1898,62 @@ sub override_list ...@@ -1792,50 +1898,62 @@ sub override_list
<tbody> <tbody>
EOF EOF
# Body # Body
foreach my $oid (@overrides) { foreach my $poid (@poolOverrides) {
my $override; my $poolOverride;
# If we can't get the override, just skip it # If we can't get the pool override, just skip it
if (!defined($override = getOverride($oid))) { if (!defined($poolOverride = getPoolOverride($poid))) {
next; next;
} }
my $oidEscaped = uri_escape($override->{'ID'}); my $poolOverrideFriendlyNameEncoded = encode_entities(prettyUndef($poolOverride->{'FriendlyName'}));
my $poolOverridePoolNameEncoded = encode_entities(prettyUndef($poolOverride->{'PoolName'}));
my $poolOverrideUsernameEncoded = encode_entities(prettyUndef($poolOverride->{'Username'}));
my $poolOverrideIPAddressEncoded = encode_entities(prettyUndef($poolOverride->{'IPAddress'}));
my $poolOverrideIPNATAddressEncoded = encode_entities(prettyUndef($poolOverride->{'IPNATAddress'}));
my $poolOverrideExpiresStr = encode_entities(
($poolOverride->{'Expires'} > 0) ?
DateTime->from_epoch( epoch => $poolOverride->{'Expires'} )->iso8601() : '-never-'
);
my $friendlyNameEncoded = prettyUndef(encode_entities($override->{'FriendlyName'})); my $poolOverrideTrafficClassStr = "-undef-";
my $usernameEncoded = prettyUndef(encode_entities($override->{'Username'})); if (defined($poolOverride->{'TrafficClassID'})) {
my $poolNameEncoded = prettyUndef(encode_entities($override->{'PoolName'})); my $trafficClass = getTrafficClass($poolOverride->{'TrafficClassID'});
my $ipAddress = prettyUndef($override->{'IPAddress'}); $poolOverrideTrafficClassStr = encode_entities($trafficClass->{'Name'});
my $expiresStr = ($override->{'Expires'} > 0) ? }
DateTime->from_epoch( epoch => $override->{'Expires'} )->iso8601() : '-never-';
my $classStr = prettyUndef(getTrafficClassName($override->{'ClassID'})); my $poolOverrideCIRStr = encode_entities(
my $cirStr = sprintf('%s/%s',prettyUndef($override->{'TrafficLimitTx'}),prettyUndef($override->{'TrafficLimitRx'})); sprintf('%s/%s',prettyUndef($poolOverride->{'TxCIR'}),prettyUndef($poolOverride->{'RxCIR'}))
my $limitStr = sprintf('%s/%s',prettyUndef($override->{'TrafficLimitTxBurst'}), );
prettyUndef($override->{'TrafficLimitRxBurst'})); my $poolOverrideLimitStr = encode_entities(
sprintf('%s/%s',prettyUndef($poolOverride->{'TxLimit'}),prettyUndef($poolOverride->{'RxLimit'}))
);
my $urlPoolOverrideEdit = sprintf('/limits/pool-override-edit?poid=%s',encode_entities($poolOverride->{'ID'}));
my $urlPoolOverrideRemove = sprintf('/limits/pool-override-remove?poid=%s',encode_entities($poolOverride->{'ID'}));
$content .= <<EOF; $content .= <<EOF;
<tr> <tr>
<td></td> <td></td>
<td>$friendlyNameEncoded</td> <td>$poolOverrideFriendlyNameEncoded</td>
<td>$poolNameEncoded</td> <td>$poolOverridePoolNameEncoded</td>
<td>$usernameEncoded</td> <td>$poolOverrideUsernameEncoded</td>
<td>$ipAddress</td> <td>$poolOverrideIPAddressEncoded</td>
<td>$expiresStr</td> <td>$poolOverrideIPNATAddressEncoded</td>
<td>$poolOverrideExpiresStr</td>
<td><span class="glyphicon glyphicon-arrow-right" /></td> <td><span class="glyphicon glyphicon-arrow-right" /></td>
<td>$classStr</td> <td class="align-center">$poolOverrideTrafficClassStr</td>
<td>$cirStr</td> <td class="align-center">$poolOverrideCIRStr</td>
<td>$limitStr</td> <td class="align-center">$poolOverrideLimitStr</td>
<td> <td>
<a href="/limits/override-edit?oid=$oidEscaped"><span class="glyphicon glyphicon-wrench" /></a> <a href="$urlPoolOverrideEdit"><span class="glyphicon glyphicon-wrench" /></a>
<a href="/limits/override-remove?oid=$oidEscaped"><span class="glyphicon glyphicon-remove" /></a> <a href="$urlPoolOverrideRemove"><span class="glyphicon glyphicon-remove" /></a>
</td> </td>
</tr> </tr>
EOF EOF
} }
# No results # No results
if (!@overrides) { if (!@poolOverrides) {
$content .=<<EOF; $content .=<<EOF;
<tr class="info"> <tr class="info">
<td colspan="11"><p class="text-center">No Results</p></td> <td colspan="11"><p class="text-center">No Results</p></td>
...@@ -1854,8 +1972,9 @@ EOF ...@@ -1854,8 +1972,9 @@ EOF
} }
# Add/edit action # Add/edit action
sub override_addedit sub pool_override_addedit
{ {
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
...@@ -1869,17 +1988,17 @@ sub override_addedit ...@@ -1869,17 +1988,17 @@ sub override_addedit
# Items for our form... # Items for our form...
my @formElements = qw( my @formElements = qw(
FriendlyName FriendlyName
PoolName Username IPAddress PoolName Username IPAddress IPNATAddress IPNATInbound
ClassID TrafficClassID
TrafficLimitTx TrafficLimitTxBurst TxCIR TxLimit
TrafficLimitRx TrafficLimitRxBurst RxCIR RxLimit
Expires inputExpires.modifier Expires inputExpires.modifier
Notes Notes
); );
my @formElementCheckboxes = qw( my @formElementCheckboxes = qw(
ClassID TrafficClassID
TrafficLimitTx TrafficLimitTxBurst TxCIR TxLimit
TrafficLimitRx TrafficLimitRxBurst RxCIR RxLimit
); );
# Expires modifier options # Expires modifier options
...@@ -1893,19 +2012,20 @@ sub override_addedit ...@@ -1893,19 +2012,20 @@ sub override_addedit
# Title of the form, by default its an add form # Title of the form, by default its an add form
my $formType = "Add"; my $formType = "Add";
my $formNoEdit = ""; my $formNoEdit = "";
my $checkboxNoEdit = "";
# Form data # Form data
my $formData; my $formData;
# If we have a override, this is where its kept # If we have a pool override, this is where its kept
my $override; my $poolOverride;
# Grab query params # Grab query params
my $queryParams = parseURIQuery($request); my $queryParams = parseURIQuery($request);
# If we have a override ID, pull in the override # If we have a pool override ID, pull in the pool override
if (defined($queryParams->{'oid'})) { if (defined($queryParams->{'poid'})) {
# Check if we can grab the override # Check if we can grab the pool override
if (!defined($override = getOverride($queryParams->{'oid'}->{'value'}))) { if (!defined($poolOverride = getPoolOverride($queryParams->{'poid'}->{'value'}))) {
return (HTTP_TEMPORARY_REDIRECT,"limits/override-list"); return (HTTP_TEMPORARY_REDIRECT,"limits/pool-override-list");
} }
} }
...@@ -1917,7 +2037,7 @@ sub override_addedit ...@@ -1917,7 +2037,7 @@ sub override_addedit
# If user pressed cancel, redirect # If user pressed cancel, redirect
if (defined($form->{'cancel'})) { if (defined($form->{'cancel'})) {
# Redirects to default page # Redirects to default page
return (HTTP_TEMPORARY_REDIRECT,'limits/override-list'); return (HTTP_TEMPORARY_REDIRECT,'/limits/pool-override-list');
} }
# Transform form into form data # Transform form into form data
...@@ -1927,24 +2047,25 @@ sub override_addedit ...@@ -1927,24 +2047,25 @@ sub override_addedit
# Set form type if its edit # Set form type if its edit
if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") { if (defined($form->{'submit'}) && $form->{'submit'}->{'value'} eq "Edit") {
# Check override exists # Check pool override exists
if (!defined($override)) { if (!defined($poolOverride)) {
return (HTTP_TEMPORARY_REDIRECT,'limits/override-list'); return (HTTP_TEMPORARY_REDIRECT,'/limits/pool-override-list');
} }
$formData->{'ID'} = $override->{'ID'}; $formData->{'ID'} = $poolOverride->{'ID'};
$formType = "Edit"; $formType = "Edit";
$formNoEdit = "readonly"; $formNoEdit = "readonly";
$checkboxNoEdit = "disabled";
} }
# A GET would indicate that a override ID was passed normally # A GET would indicate that a pool override ID was passed normally
} elsif ($request->method eq "GET") { } elsif ($request->method eq "GET") {
# We need a override # We need a pool override
if (defined($override)) { if (defined($poolOverride)) {
# Setup form data from override # Setup form data from pool override
foreach my $key (@formElements) { foreach my $key (@formElements) {
$formData->{$key} = $override->{$key}; $formData->{$key} = $poolOverride->{$key};
} }
# Setup our checkboxes # Setup our checkboxes
foreach my $checkbox (@formElementCheckboxes) { foreach my $checkbox (@formElementCheckboxes) {
...@@ -1955,10 +2076,11 @@ sub override_addedit ...@@ -1955,10 +2076,11 @@ sub override_addedit
$formType = "Edit"; $formType = "Edit";
$formNoEdit = "readonly"; $formNoEdit = "readonly";
$checkboxNoEdit = "disabled";
# Woops ... no query string? # Woops ... no query string?
} elsif (keys %{$queryParams} > 0) { } elsif (keys %{$queryParams} > 0) {
return (HTTP_TEMPORARY_REDIRECT,'limits/override-list'); return (HTTP_TEMPORARY_REDIRECT,'/limits/pool-override-list');
} }
} }
...@@ -1970,59 +2092,93 @@ sub override_addedit ...@@ -1970,59 +2092,93 @@ sub override_addedit
push(@errors,"Friendly name must be specified"); push(@errors,"Friendly name must be specified");
} }
# Make sure we have at least a pool name, username or IP address # Check the pool name is valid if it was specified
my $poolName = isUsername($formData->{'PoolName'}); my $poolName;
my $username = isUsername($formData->{'Username'}); if (defined($formData->{'PoolName'}) && $formData->{'PoolName'} ne "") {
my $ipAddress = isIP($formData->{'IPAddress'}); if (!defined($poolName = isUsername($formData->{'PoolName'},ISUSERNAME_ALLOW_ATSIGN))) {
push(@errors,"Pool name is not valid");
}
}
# Next check the username
my $username;
if (defined($formData->{'Username'}) && $formData->{'Username'} ne "") {
if (!defined($username = isUsername($formData->{'Username'},ISUSERNAME_ALLOW_ATSIGN))) {
push(@errors,"Username is not valid");
}
}
# Then the IP
my $ipAddress;
if (defined($formData->{'IPAddress'}) && $formData->{'IPAddress'} ne "") {
if (!defined($ipAddress = isIPv46CIDR($formData->{'IPAddress'}))) {
push(@errors,"IP address is not valid");
}
}
# And NAT
my $ipNATAddress;
if (defined($formData->{'IPNATAddress'}) && $formData->{'IPNATAddress'} ne "") {
if (!defined($ipNATAddress = isIPv46($formData->{'IPNATAddress'}))) {
push(@errors,"NAT address is not valid");
}
}
my $ipNATInbound;
if (defined($formData->{'IPNATInbound'}) && $formData->{'IPNATInbound'} ne "") {
if (defined($ipNATAddress)) {
$ipNATInbound = "yes";
} else {
push(@errors,"Cannot NAT inbound traffic if no NAT address is set");
}
}
# Then confirm we have at least one of the above
if (!defined($poolName) && !defined($username) && !defined($ipAddress)) { if (!defined($poolName) && !defined($username) && !defined($ipAddress)) {
push(@errors,"A pool name and/or IP address and/or Username must be specified"); push(@errors,"At least a valid pool name, username or IP address must be specified to match");
} }
# If the traffic class is ticked, process it # If the traffic class is ticked, process it
my $classID; my $trafficClassID;
if (defined($formData->{'inputClassID.enabled'})) { if (defined($formData->{'inputTrafficClassID.enabled'})) {
if (!defined($classID = isTrafficClassIDValid($formData->{'ClassID'}))) { if (!defined($trafficClassID = isTrafficClassIDValid($formData->{'TrafficClassID'}))) {
push(@errors,"Traffic class is not valid"); push(@errors,"Traffic class is not valid");
} }
} }
# Check traffic limits # Check traffic limits
my $trafficLimitTx; my $txCIR;
if (defined($formData->{'inputTrafficLimitTx.enabled'})) { if (defined($formData->{'inputTxCIR.enabled'})) {
if (!defined($trafficLimitTx = isNumber($formData->{'TrafficLimitTx'}))) { if (!defined($txCIR = isNumber($formData->{'TxCIR'}))) {
push(@errors,"Download CIR is not valid"); push(@errors,"Download CIR is not valid");
} }
} }
my $trafficLimitTxBurst; my $txLimit;
if (defined($formData->{'inputTrafficLimitTxBurst.enabled'})) { if (defined($formData->{'inputTxLimit.enabled'})) {
if (!defined($trafficLimitTxBurst = isNumber($formData->{'TrafficLimitTxBurst'}))) { if (!defined($txLimit = isNumber($formData->{'TxLimit'}))) {
push(@errors,"Download limit is not valid"); push(@errors,"Download limit is not valid");
} }
} }
# Check TrafficLimitRx # Check RxCIR
my $trafficLimitRx; my $rxCIR;
if (defined($formData->{'inputTrafficLimitRx.enabled'})) { if (defined($formData->{'inputRxCIR.enabled'})) {
if (!defined($trafficLimitRx = isNumber($formData->{'TrafficLimitRx'}))) { if (!defined($rxCIR = isNumber($formData->{'RxCIR'}))) {
push(@errors,"Upload CIR is not valid"); push(@errors,"Upload CIR is not valid");
} }
} }
my $trafficLimitRxBurst; my $rxLimit;
if (defined($formData->{'inputTrafficLimitRxBurst.enabled'})) { if (defined($formData->{'inputRxLimit.enabled'})) {
if (!defined($trafficLimitRxBurst = isNumber($formData->{'TrafficLimitRxBurst'}))) { if (!defined($rxLimit = isNumber($formData->{'RxLimit'}))) {
push(@errors,"Upload limit is not valid"); push(@errors,"Upload limit is not valid");
} }
} }
# Check that we actually have something to override # Check that we actually have something to pool override
if ( if (
!defined($classID) && !defined($trafficClassID) &&
!defined($trafficLimitTx) && !defined($trafficLimitTxBurst) && !defined($txCIR) && !defined($txLimit) &&
!defined($trafficLimitRx) && !defined($trafficLimitRxBurst) !defined($rxCIR) && !defined($rxLimit)
) { ) {
push(@errors,"Something must be specified to override"); push(@errors,"Something must be specified to override");
} }
my $expires = 0; my $expires = 0;
if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") { if (defined($formData->{'Expires'}) && $formData->{'Expires'} ne "") {
if (!defined($expires = isNumber($formData->{'Expires'}))) { if (!defined($expires = isNumber($formData->{'Expires'},ISNUMBER_ALLOW_ZERO))) {
push(@errors,"Expires value is not valid"); push(@errors,"Expires value is not valid");
# Check the modifier # Check the modifier
} else { } else {
...@@ -2053,20 +2209,21 @@ sub override_addedit ...@@ -2053,20 +2209,21 @@ sub override_addedit
# Grab notes # Grab notes
my $notes = $formData->{'Notes'}; my $notes = $formData->{'Notes'};
# If there are no errors we need to push this override # If there are no errors we need to push this pool override
if (!@errors && $request->method eq "POST") { if (!@errors && $request->method eq "POST") {
# Build override # Build pool override
my $overrideData = { my $poolOverrideData = {
'FriendlyName' => $friendlyName, 'FriendlyName' => $friendlyName,
'PoolName' => $poolName, 'PoolName' => $poolName,
'Username' => $username, 'Username' => $username,
'IPAddress' => $ipAddress, 'IPAddress' => $ipAddress,
# 'IPNATAddress' => $ipNATAddress,
# 'GroupID' => 1, # 'GroupID' => 1,
'ClassID' => $classID, 'TrafficClassID' => $trafficClassID,
'TrafficLimitTx' => $trafficLimitTx, 'TxCIR' => $txCIR,
'TrafficLimitTxBurst' => $trafficLimitTxBurst, 'TxLimit' => $txLimit,
'TrafficLimitRx' => $trafficLimitRx, 'RxCIR' => $rxCIR,
'TrafficLimitRxBurst' => $trafficLimitRxBurst, 'RxLimit' => $rxLimit,
'Expires' => $expires, 'Expires' => $expires,
'Notes' => $notes, 'Notes' => $notes,
}; };
...@@ -2074,28 +2231,30 @@ sub override_addedit ...@@ -2074,28 +2231,30 @@ sub override_addedit
# Check if this is an add or edit # Check if this is an add or edit
my $cEvent; my $cEvent;
if ($formType eq "Add") { if ($formType eq "Add") {
$cEvent = "override_add"; $cEvent = "pool_override_add";
} else { } else {
$overrideData->{'ID'} = $formData->{'ID'}; $poolOverrideData->{'ID'} = $formData->{'ID'};
$cEvent = "override_change"; $cEvent = "pool_override_change";
} }
$kernel->post("configmanager" => $cEvent => $overrideData); $kernel->post("configmanager" => $cEvent => $poolOverrideData);
$logger->log(LOG_INFO,"[WEBSERVER/OVERRIDE/ADD] Pool: %s, User: %s, IP: %s, Group: %s, Class: %s, Limits: %s/%s, ". $logger->log(LOG_INFO,"[WEBSERVER/POOL-OVERRIDE/ADD] Pool: %s, User: %s, IP: %s, NAT: %s (inbound: %s), Group: %s, Class: %s, Limits: %s/%s, ".
"Burst: %s/%s", "Burst: %s/%s",
prettyUndef($poolName), prettyUndef($poolName),
prettyUndef($username), prettyUndef($username),
prettyUndef($ipAddress), prettyUndef($ipAddress),
prettyUndef($ipNATAddress),
prettyUndef($ipNATInbound),
"", "",
prettyUndef($classID), prettyUndef($trafficClassID),
prettyUndef($trafficLimitTx), prettyUndef($txCIR),
prettyUndef($trafficLimitRx), prettyUndef($rxCIR),
prettyUndef($trafficLimitTxBurst), prettyUndef($txLimit),
prettyUndef($trafficLimitRxBurst) prettyUndef($rxLimit)
); );
return (HTTP_TEMPORARY_REDIRECT,'limits/override-list'); return (HTTP_TEMPORARY_REDIRECT,'/limits/pool-override-list');
} }
} }
...@@ -2113,28 +2272,31 @@ sub override_addedit ...@@ -2113,28 +2272,31 @@ sub override_addedit
# Form header # Form header
$content .=<<EOF; $content .=<<EOF;
<legend>$formType Override</legend> <legend>$formType Pool Override</legend>
<form role="form" method="post"> <form role="form" method="post">
EOF EOF
# Spit out errors if we have any # Spit out errors if we have any
if (@errors > 0) { if (@errors > 0) {
foreach my $error (@errors) { foreach my $error (@errors) {
$content .= '<div class="alert alert-danger">'.$error.'</div>'; $content .= '<div class="alert alert-danger">'.encode_entities($error).'</div>';
} }
} }
# Generate traffic class list # Generate traffic class list
my $trafficClasses = getTrafficClasses(); my @trafficClasses = sort(getTrafficClasses());
my $trafficClassStr = ""; my $trafficClassStr = "";
foreach my $classID (sort keys %{$trafficClasses}) { foreach my $trafficClassID (@trafficClasses) {
my $trafficClass = getTrafficClass($trafficClassID);
# Process selections nicely # Process selections nicely
my $selected = ""; my $selected = "";
if ($formData->{'ClassID'} ne "" && $formData->{'ClassID'} eq $classID) { if ($formData->{'TrafficClassID'} ne "" && $formData->{'TrafficClassID'} eq $trafficClassID) {
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$trafficClassStr .= '<option value="'.$classID.'" '.$selected.'>'.$trafficClasses->{$classID}.'</option>'; $trafficClassStr .= '<option value="'.$trafficClassID.'" '.$selected.'>'.encode_entities($trafficClass->{'Name'}).
'</option>';
} }
# Generate expires modifiers list # Generate expires modifiers list
...@@ -2150,8 +2312,8 @@ EOF ...@@ -2150,8 +2312,8 @@ EOF
$selected = "selected"; $selected = "selected";
} }
# And build the options # And build the options
$expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.$expiresModifiers->{$expireModifier}. $expiresModifierStr .= '<option value="'.$expireModifier.'" '.$selected.'>'.
'</option>'; encode_entities($expiresModifiers->{$expireModifier}).'</option>';
} }
# Blank expires if its 0 # Blank expires if its 0
...@@ -2202,13 +2364,24 @@ EOF ...@@ -2202,13 +2364,24 @@ EOF
</div> </div>
</div> </div>
</div> </div>
<div class="form-group">
<label for="IPNATAddress" class="col-md-2 control-label">NAT Address</label>
<div class="row">
<div class="col-md-4">
<input name="IPNATAddress" type="text" placeholder="- - - override not implemented - - -" class="form-control"
value="$formData->{'IPNATAddress'}" $formNoEdit />
<input name="IPNATInbound" type="checkbox" $formData->{'IPNATInbound'} $checkboxNoEdit /> NAT Inbound
</div>
</div>
</div>
<div class="form-group"> <div class="form-group">
<label for="ClassID" class="col-md-2 control-label">Traffic Class</label> <label for="TrafficClassID" class="col-md-2 control-label">Traffic Class</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<input name="inputClassID.enabled" type="checkbox" $formData->{'inputClassID.enabled'}/> Override <input name="inputTrafficClassID.enabled" type="checkbox" $formData->{'inputTrafficClassID.enabled'}/>
<select name="ClassID" class="form-control"> Override
<select name="TrafficClassID" class="form-control">
$trafficClassStr $trafficClassStr
</select> </select>
</div> </div>
...@@ -2216,14 +2389,14 @@ EOF ...@@ -2216,14 +2389,14 @@ EOF
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTx" class="col-md-2 control-label">Download CIR</label> <label for="TxCIR" class="col-md-2 control-label">Download CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<input name="inputTrafficLimitTx.enabled" type="checkbox" $formData->{'inputTrafficLimitTx.enabled'} /> <input name="inputTxCIR.enabled" type="checkbox" $formData->{'inputTxCIR.enabled'} />
Override Override
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTx" type="text" placeholder="Download CIR" class="form-control" <input name="TxCIR" type="text" placeholder="Download CIR" class="form-control"
value="$formData->{'TrafficLimitTx'}" /> value="$formData->{'TxCIR'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -2231,14 +2404,14 @@ EOF ...@@ -2231,14 +2404,14 @@ EOF
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitTxBurst" class="col-md-2 control-label">Download Limit</label> <label for="TxLimit" class="col-md-2 control-label">Download Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<input name="inputTrafficLimitTxBurst.enabled" type="checkbox" <input name="inputTxLimit.enabled" type="checkbox"
$formData->{'inputTrafficLimitTxBurst.enabled'}/> Override $formData->{'inputTxLimit.enabled'}/> Override
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitTxBurst" type="text" placeholder="Download Limit" class="form-control" <input name="TxLimit" type="text" placeholder="Download Limit" class="form-control"
value="$formData->{'TrafficLimitTxBurst'}" /> value="$formData->{'TxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -2246,28 +2419,28 @@ EOF ...@@ -2246,28 +2419,28 @@ EOF
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="inputTrafficLimitRx" class="col-md-2 control-label">Upload CIR</label> <label for="inputRxCIR" class="col-md-2 control-label">Upload CIR</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<input name="inputTrafficLimitRx.enabled" type="checkbox" <input name="inputRxCIR.enabled" type="checkbox"
$formData->{'inputTrafficLimitRx.enabled'}/> Override $formData->{'inputRxCIR.enabled'}/> Override
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRx" type="text" placeholder="Upload CIR" class="form-control" <input name="RxCIR" type="text" placeholder="Upload CIR" class="form-control"
value="$formData->{'TrafficLimitRx'}" /> value="$formData->{'RxCIR'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="TrafficLimitRxBurst" class="col-md-2 control-label">Upload Limit</label> <label for="RxLimit" class="col-md-2 control-label">Upload Limit</label>
<div class="row"> <div class="row">
<div class="col-md-3"> <div class="col-md-3">
<input name="inputTrafficLimitRxBurst.enabled" type="checkbox" <input name="inputRxLimit.enabled" type="checkbox"
$formData->{'inputTrafficLimitRxBurst.enabled'}/> Override $formData->{'inputRxLimit.enabled'}/> Override
<div class="input-group"> <div class="input-group">
<input name="TrafficLimitRxBurst" type="text" placeholder="Upload Limit" class="form-control" <input name="RxLimit" type="text" placeholder="Upload Limit" class="form-control"
value="$formData->{'TrafficLimitRxBurst'}" /> value="$formData->{'RxLimit'}" />
<span class="input-group-addon">Kbps<span> <span class="input-group-addon">Kbps<span>
</div> </div>
</div> </div>
...@@ -2308,8 +2481,9 @@ EOF ...@@ -2308,8 +2481,9 @@ EOF
} }
# Remove action # Remove action
sub override_remove sub pool_override_remove
{ {
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
...@@ -2320,26 +2494,26 @@ sub override_remove ...@@ -2320,26 +2494,26 @@ sub override_remove
# Pull in GET # Pull in GET
my $queryParams = parseURIQuery($request); my $queryParams = parseURIQuery($request);
# We need a key first of all... # We need a key first of all...
if (!defined($queryParams->{'oid'})) { if (!defined($queryParams->{'poid'})) {
$content = <<EOF; $content = <<EOF;
<div class="alert alert-danger text-center"> <div class="alert alert-danger text-center">
No override oid in query string! No pool override oid in query string!
</div> </div>
EOF EOF
goto END; goto END;
} }
# Grab the override # Grab the pool override
my $override = getOverride($queryParams->{'oid'}->{'value'}); my $poolOverride = getPoolOverride($queryParams->{'poid'}->{'value'});
# Make the oid safe for HTML # Make the oid safe for HTML
my $encodedID = encode_entities($queryParams->{'oid'}->{'value'}); my $encodedPoolOverrideID = encode_entities($queryParams->{'poid'}->{'value'});
# Make sure the oid was valid... we would have an override now if it was # Make sure the oid was valid... we would have an pool override now if it was
if (!defined($override)) { if (!defined($poolOverride)) {
$content = <<EOF; $content = <<EOF;
<div class="alert alert-danger text-center"> <div class="alert alert-danger text-center">
Invalid override oid "$encodedID"! Invalid pool override oid "$encodedPoolOverrideID"!
</div> </div>
EOF EOF
goto END; goto END;
...@@ -2352,19 +2526,19 @@ EOF ...@@ -2352,19 +2526,19 @@ EOF
# Check if its a success # Check if its a success
if ($form->{'confirm'}->{'value'} eq "Yes") { if ($form->{'confirm'}->{'value'} eq "Yes") {
# Post the removal # Post the removal
$kernel->post("configmanager" => "override_remove" => $override->{'ID'}); $kernel->post("configmanager" => "pool_override_remove" => $poolOverride->{'ID'});
} }
return (HTTP_TEMPORARY_REDIRECT,'limits/override-list'); return (HTTP_TEMPORARY_REDIRECT,'/limits/pool-override-list');
} }
# Make the friendly name HTML safe # Make the friendly name HTML safe
my $encodedFriendlyName = encode_entities($override->{'FriendlyName'}); my $encodedPoolOverrideFriendlyName = encode_entities($poolOverride->{'FriendlyName'});
# Build our confirmation dialog # Build our confirmation dialog
$content .= <<EOF; $content .= <<EOF;
<div class="alert alert-danger"> <div class="alert alert-danger">
Are you very sure you wish to remove override "$encodedFriendlyName"? Are you very sure you wish to remove pool override "$encodedPoolOverrideFriendlyName"?
</div> </div>
<form role="form" method="post"> <form role="form" method="post">
<input type="submit" class="btn btn-primary" name="confirm" value="Yes" /> <input type="submit" class="btn btn-primary" name="confirm" value="Yes" />
......
# OpenTrafficShaper webserver module: index page # OpenTrafficShaper webserver module: index page
# Copyright (C) 2007-2014, AllWorldIT # Copyright (C) 2007-2023, AllWorldIT
# #
# This program is free software: you can redistribute it and/or modify # 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 # it under the terms of the GNU General Public License as published by
...@@ -41,6 +41,7 @@ our (@ISA,@EXPORT,@EXPORT_OK); ...@@ -41,6 +41,7 @@ our (@ISA,@EXPORT,@EXPORT_OK);
); );
sub _catchall sub _catchall
{ {
my ($kernel,$globals,$client_session_id,$request) = @_; my ($kernel,$globals,$client_session_id,$request) = @_;
...@@ -76,7 +77,7 @@ sub _catchall ...@@ -76,7 +77,7 @@ sub _catchall
} }
# Stat file first of all # Stat file first of all
my $stat = stat($filename); my $stat = stat($filename);
if (!$stat) { if (!$stat) {
$logger->log(LOG_WARN,"[WEBSERVER/STATIC] Unable to stat '%s': %s",$resource,$!); $logger->log(LOG_WARN,"[WEBSERVER/STATIC] Unable to stat '%s': %s",$resource,$!);
return; return;
...@@ -107,7 +108,7 @@ sub _catchall ...@@ -107,7 +108,7 @@ sub _catchall
$response->header('Last-Modified', HTTP::Date::time2str($stat->mtime)); $response->header('Last-Modified', HTTP::Date::time2str($stat->mtime));
# Open file handle # Open file handle
if (!open(FH, "< $filename")) { if (!open(FH, "< $filename")) {
$logger->log(LOG_WARN,"[WEBSERVER/STATIC] Unable to open '%s': %s",$resource,$!); $logger->log(LOG_WARN,"[WEBSERVER/STATIC] Unable to open '%s': %s",$resource,$!);
} }
# Set to binary mode # Set to binary mode
...@@ -124,8 +125,10 @@ sub _catchall ...@@ -124,8 +125,10 @@ sub _catchall
# Set content # Set content
$response->content($buffer); $response->content($buffer);
return $response; return $response;
} }
1; 1;
# vim: ts=4