Files
jami-daemon/bin/winmain.cpp
Andreas Traczyk b0e223ff3c misc: add an API to set the client app's resource directory path
This approach allows client apps that install ringtones (and any other resources) in custom locations to specify that directory at runtime. This will take precedence over the build time data directory supplied for platforms which install to a fixed path.

https: //git.jami.net/savoirfairelinux/jami-client-qt/-/issues/1619
Change-Id: I81616f79196e645a5ad677d6956be6a2ffcd976a
2024-06-04 09:57:32 -04:00

231 lines
5.6 KiB
C++

/*
* Copyright (C) 2004-2024 Savoir-faire Linux Inc.
*
* Author: Edric Milaret <edric.ladent-milaret@savoirfairelinux.com>
*
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include <iostream>
#include <thread>
#include <cstring>
#include <signal.h>
#include <getopt.h>
#include <string>
#include <filesystem>
#include "jami.h"
#include "callmanager_interface.h"
#include "configurationmanager_interface.h"
#include "presencemanager_interface.h"
#ifdef ENABLE_VIDEO
#include "videomanager_interface.h"
#endif
#include "fileutils.h"
#include <windows.h>
using namespace std::placeholders;
bool isActive = false;
static int ringFlags = 0;
bool loop = true;
static void
print_title()
{
std::cout
<< "Jami Daemon " << libjami::version()
<< ", by Savoir-faire Linux Inc. 2004-2023" << std::endl
<< "https://jami.net/" << std::endl
#ifdef ENABLE_VIDEO
<< "[Video support enabled]" << std::endl
#endif
#ifdef ENABLE_PLUGIN
<< "[Plugins support enabled]" << std::endl
#endif
<< std::endl;
}
static void
print_usage()
{
std::cout << std::endl <<
"-c, --console \t- Log in console (instead of syslog)" << std::endl <<
"-d, --debug \t- Debug mode (more verbose)" << std::endl <<
"-p, --persistent \t- Stay alive after client quits" << std::endl <<
"--auto-answer \t- Force automatic answer to incoming calls" << std::endl <<
"-h, --help \t- Print help" << std::endl;
}
// Parse command line arguments, setting debug options or printing a help
// message accordingly.
// returns true if we should quit (i.e. help was printed), false otherwise
static bool
parse_args(int argc, char *argv[], bool& persistent)
{
int consoleFlag = false;
int debugFlag = false;
int helpFlag = false;
int versionFlag = false;
int autoAnswer = false;
const struct option long_options[] = {
/* These options set a flag. */
{"debug", no_argument, NULL, 'd'},
{"console", no_argument, NULL, 'c'},
{"persistent", no_argument, NULL, 'p'},
{"help", no_argument, NULL, 'h'},
{"version", no_argument, NULL, 'v'},
{"auto-answer", no_argument, &autoAnswer, true},
{0, 0, 0, 0} /* Sentinel */
};
while (true) {
/* getopt_long stores the option index here. */
int option_index = 0;
auto c = getopt_long(argc, argv, "dcphv", long_options, &option_index);
// end of the options
if (c == -1)
break;
switch (c) {
case 'd':
debugFlag = true;
break;
case 'c':
consoleFlag = true;
break;
case 'p':
persistent = true;
break;
case 'h':
case '?':
helpFlag = true;
break;
case 'v':
versionFlag = true;
break;
default:
break;
}
}
if (helpFlag) {
print_usage();
return true;
}
if (versionFlag) {
// We've always print the title/version, so we can just exit
return true;
}
if (consoleFlag)
ringFlags |= libjami::LIBJAMI_FLAG_CONSOLE_LOG;
if (debugFlag)
ringFlags |= libjami::LIBJAMI_FLAG_DEBUG;
if (autoAnswer)
ringFlags |= libjami::LIBJAMI_FLAG_AUTOANSWER;
return false;
}
void
IncomingCall(const std::string& accountId,
const std::string& callId, const std::string& message)
{
(void) accountId;
(void) message;
if (not isActive) {
libjami::accept(callId);
isActive = true;
} else
libjami::refuse(callId);
}
static int
run()
{
using SharedCallback = std::shared_ptr<libjami::CallbackWrapperBase>;
libjami::init(static_cast<libjami::InitFlag>(ringFlags));
std::map<std::string, SharedCallback> callHandlers;
callHandlers.insert(libjami::exportable_callback<libjami::CallSignal::IncomingCall>
(std::bind(&IncomingCall, _1, _2, _3)));
registerSignalHandlers(callHandlers);
if (!libjami::start())
return -1;
while (loop) {
Sleep(1000); // milliseconds
}
libjami::fini();
return 0;
}
static void
interrupt()
{
loop = false;
}
static void
signal_handler(int code)
{
std::cerr << "Caught signal " << code
<< ", terminating..." << std::endl;
// Unset signal handlers
signal(SIGINT, SIG_DFL);
signal(SIGTERM, SIG_DFL);
interrupt();
}
int
main(int argc, char *argv [])
{
// Set the program's directory path as the resource directory path.
std::filesystem::path programPath(argv[0]);
jami::fileutils::set_resource_dir_path(programPath.parent_path());
print_title();
bool persistent = false;
if (parse_args(argc, argv, persistent))
return 0;
// TODO: Block signals for all threads but the main thread, decide how/if we should
// handle other signals
signal(SIGINT, signal_handler);
signal(SIGTERM, signal_handler);
return run();
}