1
0
mirror of https://github.com/mfontanini/libtins synced 2026-01-29 21:14:28 +01:00

Add TCP and Utils::resolve_hwaddress active tests

[ci skip]
This commit is contained in:
Matias Fontanini
2016-03-22 19:49:26 -07:00
parent 068e304baa
commit 7bc1ab41f7
12 changed files with 366 additions and 24 deletions

View File

@@ -34,6 +34,7 @@
using std::cout;
using std::endl;
using std::string;
using Tins::PDU;
using Tins::pdu_not_found;
@@ -46,7 +47,16 @@ ActiveTest::ActiveTest(const PacketSenderPtr& packet_sender,
}
void ActiveTest::execute() {
execute_test();
if (is_enabled()) {
execute_test();
}
else {
cout << log_prefix() << "not running as test is disabled on this platform" << endl;
}
}
string ActiveTest::log_prefix() const {
return "[" + name() + "] ";
}
bool ActiveTest::matches_packet(const PDU& pdu) const {
@@ -61,6 +71,31 @@ bool ActiveTest::matches_packet(const PDU& pdu) const {
}
}
bool ActiveTest::is_enabled() const {
return (configuration_->current_platform() & disabled_platforms_) == 0;
}
void ActiveTest::validate(PacketCapturer::PacketStorage& packets) {
string prefix = log_prefix();
size_t i = 0;
while (i < packets.size() && !matches_packet(*packets[i])) {
++i;
}
if (i == packets.size()) {
cout << prefix << "ERROR: Packet was not captured" << endl;
}
else {
try {
validate_packet(*packets[i]);
cout << prefix << "OK" << endl;
}
catch (TestFailed& ex) {
cout << prefix << "ERROR: " << ex.what() << endl;
}
packets.erase(packets.begin() + i);
}
}
const ActiveTest::PacketSenderPtr& ActiveTest::packet_sender() const {
return packet_sender_;
}
@@ -69,3 +104,6 @@ const ActiveTest::ConfigurationPtr& ActiveTest::configuration() const {
return configuration_;
}
void ActiveTest::disable_on_platform(Configuration::Platform platform) {
disabled_platforms_ |= static_cast<unsigned>(platform);
}

View File

@@ -90,23 +90,9 @@ void ActiveTestRunner::do_run() {
auto packets = capturer_.captured_packets();
cout << prefix << "Captured " << packets.size() << " packets" << endl;
for (const auto& test : tests_) {
prefix = "[" + test->name() + "] ";
size_t i = 0;
while (i < packets.size() && !test->matches_packet(*packets[i])) {
++i;
}
if (i == packets.size()) {
cout << prefix << "ERROR: Packet was not captured" << endl;
}
else {
try {
test->validate_packet(*packets[i]);
cout << prefix << "OK" << endl;
}
catch (TestFailed& ex) {
cout << prefix << "ERROR: " << ex.what() << endl;
}
packets.erase(packets.begin() + i);
if (test->is_enabled()) {
test->validate(packets);
}
}
}

View File

@@ -27,12 +27,26 @@
*
*/
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
#include <sys/param.h>
#endif
#include "configuration.h"
using std::string;
using Tins::NetworkInterface;
Configuration::Configuration() {
#ifdef _WIN32
current_platform_ = WINDOWS;
#elif defined(BSD) || defined(__FreeBSD_kernel__)
current_platform_ = BSD;
#else
current_platform_ = LINUX;
#endif
}
void Configuration::interface(const NetworkInterface& interface) {
interface_ = interface;
}
@@ -57,3 +71,6 @@ uint16_t Configuration::destination_port() const {
return destination_port_;
}
Configuration::Platform Configuration::current_platform() const {
return current_platform_;
}

View File

@@ -32,6 +32,8 @@
#include "configuration.h"
#include "active_test_runner.h"
#include "ipv4_tests.h"
#include "tcp_tests.h"
#include "utils_tests.h"
#include "tins/network_interface.h"
using std::cerr;
@@ -51,6 +53,8 @@ int main() {
ActiveTestRunner runner(config);
runner.add_test<IPv4SourceAddressTest>();
runner.add_test<IPv4FragmentationTest>();
runner.add_test<TCPSynTest>();
runner.add_test<ResolveHWAddressTest>();
if (!runner.validate_tests()) {
throw runtime_error("Test validation failed");

View File

@@ -95,10 +95,11 @@ bool PacketCapturer::callback(const PDU& pdu) {
string PacketCapturer::make_filter(const Configuration& configuration) const {
ostringstream oss;
oss << "((tcp or udp) and src port " << configuration.source_port()
<< " and dst port " << configuration.destination_port() << ") or icmp"
oss << "((tcp or udp) and (port " << configuration.source_port()
<< " or port " << configuration.destination_port() << ")) or icmp"
// Fragmentted IP packets
<< " or (ip[6:2] & 0x1fff) > 0";
<< " or (ip[6:2] & 0x1fff) > 0"
<< " or arp";
return oss.str();
}

View File

@@ -0,0 +1,87 @@
/*
* Copyright (c) 2016, Matias Fontanini
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include <random>
#include <iostream>
#include "tcp_tests.h"
#include "tins/tcp.h"
#include "tins/ip.h"
#include "tins/utils.h"
using std::string;
using std::cout;
using std::endl;
using std::random_device;
using Tins::PDU;
using Tins::TCP;
using Tins::IP;
using Tins::Utils::resolve_domain;
TCPSynTest::TCPSynTest(const PacketSenderPtr& packet_sender,
const ConfigurationPtr& configuration)
: ActiveTest(packet_sender, configuration) {
disable_on_platform(Configuration::WINDOWS);
}
string TCPSynTest::name() const {
return "tcp_syn_test";
}
void TCPSynTest::execute_test() {
random_device rnd;
target_address_ = resolve_domain("www.example.com");
sequence_number_ = static_cast<uint32_t>(rnd());
cout << log_prefix() << "Resolved target address to " << target_address_ << endl;
auto packet = IP(target_address_) / TCP(80, configuration()->source_port());
TCP& tcp = packet.rfind_pdu<TCP>();
tcp.seq(sequence_number_);
tcp.flags(TCP::SYN);
packet_sender()->send(packet);
}
void TCPSynTest::validate_packet(const PDU& pdu) {
const TCP& tcp = pdu.rfind_pdu<TCP>();
if (tcp.flags() != (TCP::SYN | TCP::ACK) && tcp.flags() != TCP::RST) {
throw TestFailed("Invalid flags received");
}
}
bool TCPSynTest::test_matches_packet(const PDU& pdu) const {
if (pdu.rfind_pdu<IP>().src_addr() != target_address_) {
return false;
}
const TCP& tcp = pdu.rfind_pdu<TCP>();
if (tcp.sport() != 80) {
return false;
}
return tcp.ack_seq() == sequence_number_ + 1;
}

View File

@@ -0,0 +1,90 @@
/*
* Copyright (c) 2016, Matias Fontanini
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include <sstream>
#include <iostream>
#include "utils_tests.h"
#include "tins/ethernetII.h"
#include "tins/arp.h"
#include "tins/utils.h"
using std::cout;
using std::endl;
using std::string;
using std::vector;
using std::ostringstream;
using Tins::PDU;
using Tins::EthernetII;
using Tins::ARP;
using Tins::IPv4Address;
using Tins::HWAddress;
using Tins::Utils::RouteEntry;
ResolveHWAddressTest::ResolveHWAddressTest(const PacketSenderPtr& packet_sender,
const ConfigurationPtr& configuration)
: ActiveTest(packet_sender, configuration) {
vector<RouteEntry> entries = Tins::Utils::route_entries();
string interface_name = configuration->interface().name();
for (const auto& entry : entries) {
if (entry.interface == interface_name && entry.gateway != "0.0.0.0") {
target_address_ = entry.gateway;
}
}
disable_on_platform(Configuration::WINDOWS);
}
string ResolveHWAddressTest::name() const {
return "resolve_hwaddress";
}
bool ResolveHWAddressTest::test_matches_packet(const PDU& pdu) const {
const ARP& arp = pdu.rfind_pdu<ARP>();
return arp.opcode() == ARP::REPLY && arp.sender_ip_addr() == target_address_;
}
void ResolveHWAddressTest::execute_test() {
cout << log_prefix() << "trying to resolve " << target_address_ << endl;
resolved_address_ = Tins::Utils::resolve_hwaddr(configuration()->interface(),
target_address_,
*packet_sender());
auto local_ip_address = configuration()->interface().ipv4_address();
auto local_hw_address = configuration()->interface().hw_address();
auto packet = ARP::make_arp_request(target_address_, local_ip_address, local_hw_address);
packet_sender()->send(packet);
}
void ResolveHWAddressTest::validate_packet(const PDU& pdu) {
const ARP& arp = pdu.rfind_pdu<ARP>();
if (arp.sender_hw_addr() != resolved_address_) {
ostringstream oss;
oss << "Expected address " << resolved_address_ << " but got " << arp.sender_hw_addr();
throw TestFailed(oss.str());
}
}