/* vim: set expandtab ts=4 sw=4: */
/*
* You may redistribute this program 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 .
*/
#include "test/BeaconFramework.h"
#include "io/FileWriter.h"
#include "memory/Allocator.h"
#include "crypto/random/Random.h"
#include "interface/Iface.h"
#include "util/events/EventBase.h"
#include "util/log/WriterLog.h"
#include "test/TestFramework.h"
#include "wire/Headers.h"
#include "wire/Ethernet.h"
#include "interface/tuntap/TUNMessageType.h"
#include "util/Hex.h"
#include "util/events/Time.h"
#include "util/events/Timeout.h"
#include
struct TwoNodes;
typedef void (RunTest)(struct TwoNodes* ctx);
struct TwoNodes
{
struct TestFramework* nodeB;
struct Iface tunB;
struct TestFramework* nodeA;
struct Iface tunA;
int messageFrom;
bool beaconsSent;
struct Timeout* checkLinkageTimeout;
struct Log* logger;
EventBase_t* base;
struct Allocator* alloc;
uint64_t startTime;
RunTest* runTest;
Identity
};
#define TUNB 2
#define TUNA 1
static Iface_DEFUN incomingTunB(Message_t* msg, struct Iface* tunB)
{
struct TwoNodes* tn = Identity_containerOf(tunB, struct TwoNodes, tunB);
uint16_t t = -1;
Err(TUNMessageType_pop(&t, msg));
Assert_true(t == Ethernet_TYPE_IP6);
Err(Message_eshift(msg, -Headers_IP6Header_SIZE));
printf("Message from TUN in node B [%s]\n", Message_bytes(msg));
tn->messageFrom = TUNB;
return NULL;
}
static Iface_DEFUN incomingTunA(Message_t* msg, struct Iface* tunA)
{
struct TwoNodes* tn = Identity_containerOf(tunA, struct TwoNodes, tunA);
uint16_t t = -1;
Err(TUNMessageType_pop(&t, msg));
Assert_true(t == Ethernet_TYPE_IP6);
Err(Message_eshift(msg, -Headers_IP6Header_SIZE));
uint8_t buff[1024];
Hex_encode(buff, 1024, Message_bytes(msg), Message_getLength(msg));
printf("Message from TUN in node A [%s] [%d] [%s]\n", Message_bytes(msg), Message_getLength(msg), buff);
tn->messageFrom = TUNA;
return NULL;
}
static void notLinkedYet(struct TwoNodes* ctx)
{
uint64_t now = Time_currentTimeMilliseconds();
if ((now - ctx->startTime) > 5000) {
Assert_failure("Failed to link in 5 seconds");
}
}
static void checkLinkage(void* vTwoNodes)
{
struct TwoNodes* ctx = Identity_check((struct TwoNodes*) vTwoNodes);
if (!ctx->beaconsSent) {
Log_debug(ctx->logger, "Linking A and B");
TestFramework_linkNodes(ctx->nodeB, ctx->nodeA, true);
ctx->beaconsSent = true;
return;
}
//Log_debug(ctx->logger, "Checking linkage");
if (SessionManager_handleCount(ctx->nodeA->nc->sm) < 1) {
notLinkedYet(ctx);
return;
}
Log_debug(ctx->logger, "A seems to be linked with B");
if (SessionManager_handleCount(ctx->nodeA->nc->sm) < 1) {
notLinkedYet(ctx);
return;
}
Log_debug(ctx->logger, "B seems to be linked with A");
Log_debug(ctx->logger, "\n\nSetup Complete\n\n");
Timeout_clearTimeout(ctx->checkLinkageTimeout);
ctx->runTest(ctx);
}
static void start(struct Allocator* alloc,
struct Log* logger,
EventBase_t* base,
struct Random* rand,
RunTest* runTest,
bool noiseA,
bool noiseB)
{
#if defined(ADDRESS_PREFIX) || defined(ADDRESS_PREFIX_BITS)
uint8_t address[16];
uint8_t publicKey[32];
uint8_t privateKeyA[32];
Key_gen(address, publicKey, privateKeyA, rand);
struct TestFramework* a =
TestFramework_setUp((char*) privateKeyA, alloc, base, rand, logger, noiseA);
uint8_t privateKeyB[32];
Key_gen(address, publicKey, privateKeyB, rand);
struct TestFramework* b =
TestFramework_setUp((char*) privateKeyB, alloc, base, rand, logger, noiseB);
#else
struct TestFramework* a =
TestFramework_setUp("\xad\x7e\xa3\x26\xaa\x01\x94\x0a\x25\xbc\x9e\x01\x26\x22\xdb\x69"
"\x4f\xd9\xb4\x17\x7c\xf3\xf8\x91\x16\xf3\xcf\xe8\x5c\x80\xe1\x4a",
alloc, base, rand, logger, noiseA);
//"publicKey": "kmzm4w0kj9bswd5qmx74nu7kusv5pj40vcsmp781j6xxgpd59z00.k",
//"ipv6": "fc41:94b5:0925:7ba9:3959:11ab:a006:367a",
struct TestFramework* b =
TestFramework_setUp("\xd8\x54\x3e\x70\xb9\xae\x7c\x41\xbc\x18\xa4\x9a\x9c\xee\xca\x9c"
"\xdc\x45\x01\x96\x6b\xbd\x7e\x76\xcf\x3a\x9f\xbc\x12\xed\x8b\xb4",
alloc, base, rand, logger, noiseB);
//"publicKey": "vz21tg07061s8v9mckrvgtfds7j2u5lst8cwl6nqhp81njrh5wg0.k",
//"ipv6": "fc1f:5b96:e1c5:625d:afde:2523:a7fa:383a",
#endif
struct TwoNodes* out = Allocator_calloc(alloc, sizeof(struct TwoNodes), 1);
Identity_set(out);
out->tunB.send = incomingTunB;
out->tunA.send = incomingTunA;
Iface_plumb(&out->tunB, b->tunIf);
Iface_plumb(&out->tunA, a->tunIf);
out->nodeB = b;
out->nodeA = a;
out->logger = logger;
out->checkLinkageTimeout = Timeout_setInterval(checkLinkage, out, 1, base, alloc);
out->base = base;
out->startTime = Time_currentTimeMilliseconds();
out->runTest = runTest;
out->alloc = alloc;
Log_debug(a->logger, "Waiting for nodes to link asynchronously...");
}
static void sendMessage(struct TwoNodes* tn,
char* message,
struct TestFramework* from,
struct TestFramework* to)
{
Message_t* msg = Message_new(64, 512, from->alloc);
uint8_t* b = NULL;
Err_assert(Message_peakBytes(&b, msg, 64));
CString_strcpy(b, message);
Err_assert(Message_truncate(msg, CString_strlen(message) + 1));
TestFramework_craftIPHeader(msg, from->ip, to->ip);
struct Iface* fromIf;
if (from == tn->nodeA) {
fromIf = &tn->tunA;
} else if (from == tn->nodeB) {
fromIf = &tn->tunB;
} else {
Assert_true(false);
}
Err_assert(TUNMessageType_push(msg, Ethernet_TYPE_IP6));
RTypes_Error_t* err = Iface_send(fromIf, msg);
if (err) {
Assert_failure("%s\n", Rffi_printError(err, tn->alloc));
}
if (to == tn->nodeA) {
Assert_true(tn->messageFrom == TUNA);
} else if (to == tn->nodeB) {
Assert_true(tn->messageFrom == TUNB);
} else {
Assert_true(false);
}
TestFramework_assertLastMessageUnaltered(tn->nodeA);
TestFramework_assertLastMessageUnaltered(tn->nodeB);
tn->messageFrom = 0;
}
static void runTest(struct TwoNodes* tn)
{
sendMessage(tn, "Hello World!", tn->nodeA, tn->nodeB);
sendMessage(tn, "Hello cjdns!", tn->nodeB, tn->nodeA);
sendMessage(tn, "send", tn->nodeA, tn->nodeB);
sendMessage(tn, "a", tn->nodeB, tn->nodeA);
sendMessage(tn, "few", tn->nodeA, tn->nodeB);
sendMessage(tn, "packets", tn->nodeB, tn->nodeA);
sendMessage(tn, "to", tn->nodeA, tn->nodeB);
sendMessage(tn, "make", tn->nodeB, tn->nodeA);
sendMessage(tn, "sure", tn->nodeA, tn->nodeB);
sendMessage(tn, "the", tn->nodeB, tn->nodeA);
sendMessage(tn, "cryptoauth", tn->nodeA, tn->nodeB);
sendMessage(tn, "can", tn->nodeB, tn->nodeA);
sendMessage(tn, "establish", tn->nodeA, tn->nodeB);
Log_debug(tn->logger, "\n\nTest passed, shutting down\n\n");
EventBase_t* base = tn->base;
Allocator_free(tn->alloc);
EventBase_endLoop(base);
}
/** Check if nodes A and C can communicate via B without A knowing that C exists. */
int BeaconFramework_test(bool noiseA, bool noiseB)
{
struct Allocator* alloc = Allocator_new(1<<22);
struct Writer* logwriter = FileWriter_new(stdout, alloc);
struct Log* logger = WriterLog_new(logwriter, alloc);
struct Random* rand = NULL;
Err_assert(Random_new(&rand, alloc, logger));
EventBase_t* base = EventBase_new(alloc);
start(Allocator_child(alloc), logger, base, rand, runTest, noiseA, noiseB);
EventBase_beginLoop(base);
Allocator_free(alloc);
return 0;
}