lokinet/daemon/main.cpp

174 lines
3.6 KiB
C++
Raw Normal View History

2018-05-27 18:03:10 +00:00
#include <llarp.h>
#include <llarp/logger.hpp>
2018-07-13 13:36:51 +00:00
#include <signal.h>
#include <getopt.h>
2018-07-20 04:50:28 +00:00
#include <string>
#include <iostream>
#ifndef _MSC_VER
#include <libgen.h>
#endif
#include "fs.hpp"
#include "config.hpp" // for ensure_config
2018-05-20 16:15:16 +00:00
#ifdef _WIN32
#define wmin(x, y) (((x) < (y)) ? (x) : (y))
#define MIN wmin
#endif
2018-05-27 18:03:10 +00:00
struct llarp_main *ctx = 0;
2018-04-30 16:14:20 +00:00
void
handle_signal(int sig)
2018-05-18 17:50:21 +00:00
{
if(ctx)
2018-05-27 18:03:10 +00:00
llarp_main_signal(ctx, sig);
2018-05-18 17:50:21 +00:00
}
int
printHelp(const char *argv0, int code = 1)
{
std::cout << "usage: " << argv0 << " [-h] [-g|-c] config.ini" << std::endl;
return code;
}
#ifdef _WIN32
int
startWinsock()
{
WSADATA wsockd;
int err;
// We used to defer starting winsock until
// we got to the iocp event loop
// but getaddrinfo(3) requires that winsock be in core already
err = ::WSAStartup(MAKEWORD(2, 2), &wsockd);
if(err)
{
perror("Failed to start Windows Sockets");
return err;
}
return 0;
}
#endif
int
main(int argc, char *argv[])
{
2018-07-20 04:50:28 +00:00
bool multiThreaded = true;
const char *singleThreadVar = getenv("LLARP_SHADOW");
if(singleThreadVar && std::string(singleThreadVar) == "1")
{
multiThreaded = false;
}
#ifdef _WIN32
if(startWinsock())
return -1;
#endif
int opt = 0;
bool genconfigOnly = false;
bool asRouter = true;
bool overWrite = false;
while((opt = getopt(argc, argv, "hgcf")) != -1)
{
switch(opt)
{
case 'h':
return printHelp(argv[0], 0);
case 'g':
genconfigOnly = true;
break;
case 'c':
genconfigOnly = true;
asRouter = false;
break;
case 'f':
overWrite = true;
break;
default:
return printHelp(argv[0]);
}
}
2018-09-19 13:15:07 +00:00
std::string conffname;
if(optind < argc)
{
// when we have an explicit filepath
fs::path fname = fs::path(argv[optind]);
fs::path basedir = fname.parent_path();
2018-09-19 13:17:15 +00:00
conffname = fname.string();
2018-09-19 13:15:07 +00:00
if(basedir.string().empty())
{
2018-10-05 07:57:48 +00:00
if(!llarp_ensure_config(fname.string().c_str(), nullptr, overWrite,
asRouter))
return 1;
2018-09-19 13:15:07 +00:00
}
else
{
std::error_code ec;
if(!fs::create_directories(basedir, ec))
{
if(ec)
{
llarp::LogError("failed to create '", basedir.string(),
"': ", ec.message());
return 1;
}
}
2018-09-19 13:15:07 +00:00
if(!llarp_ensure_config(fname.string().c_str(), basedir.string().c_str(),
overWrite, asRouter))
2018-09-19 13:15:07 +00:00
return 1;
}
}
else
{
// no explicit config file provided
#ifdef _WIN32
fs::path homedir = fs::path(getenv("APPDATA"));
#else
fs::path homedir = fs::path(getenv("HOME"));
#endif
fs::path basepath = homedir / fs::path(".lokinet");
fs::path fpath = basepath / "lokinet.ini";
std::error_code ec;
// These paths are guaranteed to exist - $APPDATA or $HOME
// so only create .lokinet/*
if(!fs::create_directory(basepath, ec))
{
2018-09-19 13:32:50 +00:00
if(ec)
{
llarp::LogError("failed to create '", basepath.string(),
"': ", ec.message());
return 1;
}
}
if(!llarp_ensure_config(fpath.string().c_str(), basepath.string().c_str(),
overWrite, asRouter))
return 1;
2018-09-19 13:15:07 +00:00
conffname = fpath.string();
}
if(genconfigOnly)
return 0;
2018-07-13 13:36:51 +00:00
2018-09-19 13:15:07 +00:00
ctx = llarp_main_init(conffname.c_str(), multiThreaded);
2018-05-27 19:13:25 +00:00
int code = 1;
if(ctx)
2018-05-27 18:03:10 +00:00
{
signal(SIGINT, handle_signal);
2018-09-19 11:57:07 +00:00
#ifndef _WIN32
2018-09-17 11:47:34 +00:00
signal(SIGHUP, handle_signal);
2018-09-19 11:57:07 +00:00
#endif
2018-05-27 19:13:25 +00:00
code = llarp_main_run(ctx);
2018-07-20 04:50:28 +00:00
llarp_main_free(ctx);
2018-05-27 18:03:10 +00:00
}
#ifdef _WIN32
::WSACleanup();
#endif
2018-08-12 17:22:29 +00:00
exit(code);
2018-05-27 19:13:25 +00:00
return code;
2017-09-28 17:02:05 +00:00
}