lokinet/daemon/main.cpp

262 lines
6.0 KiB
C++
Raw Normal View History

2018-12-12 01:47:29 +00:00
#include <config.hpp> // for ensure_config
#include <libgen.h>
2018-05-27 18:03:10 +00:00
#include <llarp.h>
2019-01-11 01:59:44 +00:00
#include <util/fs.hpp>
#include <util/logger.hpp>
2019-01-11 01:59:44 +00:00
#include <getopt.h>
2018-07-13 13:36:51 +00:00
#include <signal.h>
2019-02-02 05:51:20 +00:00
#if !defined(_WIN32) && !defined(__OpenBSD__)
2019-01-29 11:23:40 +00:00
#include <wordexp.h>
#endif
2018-12-12 01:47:29 +00:00
2018-07-20 04:50:28 +00:00
#include <string>
#include <iostream>
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)
{
2018-11-08 12:31:50 +00:00
std::cout << "usage: " << argv0 << " [-h] [-v] [-g|-c] config.ini"
<< std::endl;
return code;
}
#ifdef _WIN32
int
startWinsock()
{
WSADATA wsockd;
int err;
err = ::WSAStartup(MAKEWORD(2, 2), &wsockd);
if(err)
{
perror("Failed to start Windows Sockets");
return err;
}
return 0;
}
2018-12-14 12:50:45 +00:00
extern "C" BOOL FAR PASCAL
handle_signal_win32(DWORD fdwCtrlType)
{
UNREFERENCED_PARAMETER(fdwCtrlType);
handle_signal(SIGINT);
2018-12-23 13:29:11 +00:00
return TRUE; // probably unreachable
2018-12-14 12:50:45 +00:00
}
#endif
2019-01-29 11:23:40 +00:00
/// resolve ~ and symlinks into actual paths (so we know the real path on disk,
/// to remove assumptions and confusion with permissions)
std::string
resolvePath(std::string conffname)
{
2019-02-02 05:51:20 +00:00
// implemented in netbsd, removed downstream for security reasons
// even though it is defined by POSIX.1-2001+
#if !defined(_WIN32) && !defined(__OpenBSD__)
2019-01-29 11:23:40 +00:00
wordexp_t exp_result;
wordexp(conffname.c_str(), &exp_result, 0);
char *resolvedPath = realpath(exp_result.we_wordv[0], NULL);
if(!resolvedPath)
{
llarp::LogWarn("Can't resolve path: ", exp_result.we_wordv[0]);
return "";
}
return resolvedPath;
#else
// TODO(despair): dig through LLVM local patch set
// one of these exists deep in the bowels of LLVMSupport
return conffname; // eww, easier said than done outside of cygwin
#endif
2019-01-29 11:23:40 +00:00
}
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;
2018-12-14 12:50:45 +00:00
SetConsoleCtrlHandler(handle_signal_win32, TRUE);
#endif
#ifdef LOKINET_DEBUG
absl::SetMutexDeadlockDetectionMode(absl::OnDeadlockCycle::kAbort);
#endif
int opt = 0;
bool genconfigOnly = false;
bool asRouter = false;
bool overWrite = false;
2018-11-08 12:31:50 +00:00
while((opt = getopt(argc, argv, "hgcfrv")) != -1)
{
switch(opt)
{
2018-11-08 12:31:50 +00:00
case 'v':
SetLogLevel(llarp::eLogDebug);
llarp::LogDebug("debug logging activated");
break;
case 'h':
return printHelp(argv[0], 0);
case 'g':
genconfigOnly = true;
break;
case 'c':
genconfigOnly = true;
break;
case 'r':
asRouter = true;
break;
case 'f':
overWrite = true;
break;
default:
return printHelp(argv[0]);
}
}
2019-01-29 11:23:40 +00:00
std::string conffname; // suggestions: confFName? conf_fname?
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();
2019-01-29 11:23:40 +00:00
conffname = resolvePath(conffname);
std::error_code ec;
// llarp::LogDebug("Basedir: ", basedir);
2018-09-19 13:15:07 +00:00
if(basedir.string().empty())
{
2019-01-29 11:23:40 +00:00
// relative path to config
// does this file exist?
if(genconfigOnly)
{
if(!llarp_ensure_config(conffname.c_str(), nullptr, overWrite,
asRouter))
return 1;
}
else
{
if(!fs::exists(fname, ec))
{
llarp::LogError("Config file not found ", conffname);
return 1;
}
}
2018-09-19 13:15:07 +00:00
}
else
{
2019-01-29 11:23:40 +00:00
// absolute path to config
2018-09-19 13:15:07 +00:00
if(!fs::create_directories(basedir, ec))
{
if(ec)
{
llarp::LogError("failed to create '", basedir.string(),
"': ", ec.message());
return 1;
}
}
2019-01-29 11:23:40 +00:00
if(genconfigOnly)
{
// find or create file
if(!llarp_ensure_config(conffname.c_str(), basedir.string().c_str(),
overWrite, asRouter))
return 1;
}
else
{
// does this file exist?
if(!fs::exists(conffname, ec))
{
llarp::LogError("Config file not found ", conffname);
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";
2019-01-29 11:23:40 +00:00
// I don't think this is necessary with this condition
// conffname = resolvePath(conffname);
llarp::LogDebug("Find or create ", basepath.string());
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;
}
}
2019-01-29 11:23:40 +00:00
// if using default INI file, we're create it even if you don't ask us too
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
// this is important, can downgrade from Info though
llarp::LogInfo("Running from: ", fs::current_path().string());
llarp::LogInfo("Using config file: ", conffname);
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-11-29 13:12:28 +00:00
signal(SIGTERM, 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-12-03 02:17:36 +00:00
code = llarp_main_setup(ctx);
if(code == 0)
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
}