mirror of
https://git.jami.net/savoirfairelinux/jami-daemon.git
synced 2025-08-12 22:09:25 +08:00

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
231 lines
5.6 KiB
C++
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();
|
|
}
|