44 #include <Ice/Communicator.h>
45 #include <Ice/Initialize.h>
46 #include <Ice/LocalException.h>
47 #include <Ice/NativePropertiesAdmin.h>
48 #include <Ice/ObjectF.h>
49 #include <Ice/Process.h>
50 #include <Ice/Properties.h>
51 #include <Ice/Properties.h>
52 #include <Ice/PropertiesAdmin.h>
53 #include <Ice/PropertiesAdmin.h>
54 #include <IceUtil/Time.h>
56 #include <SimoxUtility/algorithm/string/string_tools.h>
71 #include "ArmarXCore/interface/core/Log.h"
72 #include "ArmarXCore/interface/core/Profiler.h"
77 #include "../logging/ArmarXLogBuf.h"
99 std::mutex Application::instanceMutex;
101 Ice::StringSeq Application::ProjectDependendencies;
102 std::string Application::ProjectName;
104 std::string
errormsg =
"Segmentation fault - Backtrace: \n";
119 std::array<void*, 20> array;
121 size = backtrace(array.data(), 20);
123 #pragma GCC diagnostic push
124 #pragma GCC diagnostic ignored "-Wunused-result"
126 #pragma GCC diagnostic pop
127 backtrace_symbols_fd(array.data(), size, STDERR_FILENO);
133 std::stringstream
str;
137 str <<
"Error: Abort\nBacktrace:\n";
141 str <<
"Error: signal " << sig;
154 s <<
"qtcreator -debug " << getpid();
156 res = system(
s.str().c_str());
173 Application::loadLibrariesFromProperties()
175 auto loadLibString = getProperty<std::string>(
"LoadLibraries").getValue();
176 if (!loadLibString.empty())
178 if (!libLoadingEnabled)
180 throw LocalException(
"Loading of dynamic libraries is not enabled in this application. "
181 "The Application needs to call enableLibLoading()");
183 auto entries =
armarx::Split(loadLibString,
";",
true,
true);
184 for (
auto& entry : entries)
188 std::filesystem::path path;
195 auto elements =
Split(entry,
":");
202 std::string libFileName =
"lib" + elements.at(1) +
"." +
206 std::filesystem::path testPath(libDir);
207 testPath /= libFileName;
208 if (std::filesystem::exists(testPath))
218 std::string libFileName =
"lib" + elements.at(0) +
"_" + elements.at(1) +
223 std::filesystem::path testPath(libDir);
224 testPath /= libFileName;
225 if (std::filesystem::exists(testPath))
236 <<
"' in any of the following paths: " << p.getLibraryPaths();
257 return forbidThreadCreation;
263 forbidThreadCreation =
value;
264 if (forbidThreadCreation)
266 ARMARX_INFO <<
"Thread creation with RunningTask and PeriodicTask is now forbidden in this "
271 ARMARX_INFO <<
"Thread creation with RunningTask and PeriodicTask is now allowed again in "
279 this->libLoadingEnabled = enable;
286 <<
"Command line arguments were already set: " <<
VAROUT(commandLineArguments);
287 commandLineArguments.reserve(argc);
288 for (
int i = 0; i < argc; ++i)
290 commandLineArguments.emplace_back(argv[i]);
294 const std::vector<std::string>&
297 return commandLineArguments;
317 std::unique_lock lock(instanceMutex);
325 std::unique_lock lock(instanceMutex);
338 armarXManager->setComponentIceProperties(properties);
350 armarXManager->updateComponentIceProperties(properties);
359 if (changedProperties.count(
"DisableLogging") &&
360 getProperty<bool>(
"DisableLogging").isSet())
363 armarXManager->enableLogging(!getProperty<bool>(
"DisableLogging").getValue());
366 if (changedProperties.count(
"EnableProfiling"))
368 armarXManager->enableProfiling(getProperty<bool>(
"EnableProfiling").getValue());
370 if (changedProperties.count(
"Verbosity"))
372 armarXManager->setGlobalMinimumLoggingLevel(
373 getProperty<MessageTypeT>(
"Verbosity").getValue());
421 std::cout <<
"Version: " <<
GetVersion() << std::endl;
433 const Ice::PropertyDict& env_props =
436 for (
const auto& [name,
value] : env_props)
438 const std::string short_name = name.substr(4);
439 ARMARX_INFO <<
"Setting environment variable '" << short_name <<
"' to value '" <<
value
441 setenv(short_name.c_str(),
value.c_str(), 1);
445 if (getProperty<std::uint64_t>(
"SecondsStartupDelay").isSet())
447 const auto delay = getProperty<std::uint64_t>(
"SecondsStartupDelay").getValue();
448 ARMARX_INFO <<
"startup delay: " << delay <<
" seconds";
449 std::this_thread::sleep_for(std::chrono::seconds{delay});
457 if (getProperty<std::string>(
"ApplicationName").isSet())
459 applicationName = getProperty<std::string>(
"ApplicationName").getValue();
464 const bool redirectStdout = getProperty<bool>(
"RedirectStdout").getValue();
467 std::streambuf* cout_sbuf =
nullptr;
468 std::streambuf* cerr_sbuf =
nullptr;
471 cout_sbuf = std::cout.rdbuf();
472 cerr_sbuf = std::cerr.rdbuf();
473 std::cout.rdbuf(&buf);
474 std::cerr.rdbuf(&errbuf);
481 std::cout.rdbuf(cout_sbuf);
482 std::cerr.rdbuf(cerr_sbuf);
486 int result = EXIT_SUCCESS;
488 threadPool.reset(
new ThreadPool(getProperty<unsigned int>(
"ThreadPoolSize").getValue()));
492 armarXManager =
new ArmarXManager(applicationName, communicator());
494 catch (Ice::ConnectFailedException&)
499 armarXManager->setDataPaths(getProperty<std::string>(
"DataPath").getValue());
508 callbackOnInterrupt();
512 if (applicationNetworkStats)
514 ProfilerListenerPrx profilerTopic =
515 armarXManager->getIceManager()->getTopic<ProfilerListenerPrx>(
516 armarx::Profiler::PROFILER_TOPIC_NAME);
517 applicationNetworkStats->start(profilerTopic, applicationName);
520 loadLibrariesFromProperties();
525 setup(armarXManager, properties);
530 armarXManager->shutdown();
534 result =
exec(armarXManager);
540 shutdownThread->join();
553 armarXManager->waitForShutdown();
568 std::string configFiles = communicator()->getProperties()->getProperty(
"ArmarX.Config");
574 if (!configFiles.empty() && configFiles.compare(
"1") != 0)
578 for (std::string configFile : configFileList)
580 if (!configFile.empty())
582 properties->load(configFile);
605 std::ofstream out(options.
outfile);
615 std::cout <<
"Could not write to file " << options.
outfile << std::endl;
629 this->applicationName = name;
635 return applicationName;
642 if (applicationNetworkStats)
644 applicationNetworkStats->stopTask();
647 shutdownThread.reset(
new std::thread{[
this]
650 this->armarXManager->shutdown();
659 dependencies = simox::alg::replace_all(dependencies,
"\"",
"");
662 for (
size_t i = 0; i < resultList.size(); i++)
672 ARMARX_INFO_S <<
"loading took " << (IceUtil::Time::now() - start).toMilliSeconds();
678 const std::string autodiscoverPropertyName =
"AutodiscoverPackages";
682 ARMARX_VERBOSE <<
"`" << autodiscoverPropertyName <<
"` property available.";
683 return getProperty<bool>(autodiscoverPropertyName).getValue();
689 std::vector<std::string>
701 Ice::StringSeq result = getProperty<Ice::StringSeq>(
"DefaultPackages").getValue();
702 Ice::StringSeq additional = getProperty<Ice::StringSeq>(
"AdditionalPackages").getValue();
703 result.insert(result.end(), additional.begin(), additional.end());
711 char* env_armarx_workspace = getenv(
"ARMARX_WORKSPACE");
712 char* env_armarx_default_config_dir_name = getenv(
"ARMARX_CONFIG_DIR_NAME");
714 std::filesystem::path armarx_workspace;
715 std::filesystem::path armarx_config_dir;
717 if (env_armarx_workspace)
719 armarx_workspace = std::filesystem::path(env_armarx_workspace);
723 char* home = getenv(
"HOME");
727 armarx_workspace = std::filesystem::path(home);
731 armarx_workspace =
"~/";
735 if (env_armarx_default_config_dir_name)
737 armarx_config_dir = std::filesystem::path(env_armarx_default_config_dir_name);
741 if (env_armarx_workspace)
743 armarx_config_dir =
"armarx_config";
748 armarx_config_dir =
".armarx";
754 return (armarx_workspace / armarx_config_dir).string();
758 if (env_armarx_workspace)
760 return "${ARMARX_WORKSPACE}/" + armarx_config_dir.string();
765 return "${HOME}/" + armarx_config_dir.string();
773 return armarXManager;
781 Ice::InitializationData id(initData);
786 if (
id.properties->getProperty(
"Ice.Admin.InstanceName").empty())
788 id.properties->setProperty(
"Ice.Admin.InstanceName", this->
getName());
790 if (
id.properties->getProperty(
"Ice.Admin.Endpoints").empty())
792 id.properties->setProperty(
"Ice.Admin.Endpoints",
"tcp -h 127.0.0.1");
795 if (initData.properties->getProperty(
"ArmarX.NetworkStats") ==
"1")
798 id.observer = applicationNetworkStats;
800 const auto oldFacets = initData.properties->getPropertyAsList(
"Ice.Admin.Facets");
801 if (oldFacets.empty())
803 initData.properties->setProperty(
"Ice.Admin.Facets",
"Metrics");
805 else if (std::find(oldFacets.begin(), oldFacets.end(),
"Metrics") == oldFacets.end())
807 std::stringstream
str;
809 for (
const auto& facet : oldFacets)
813 initData.properties->setProperty(
"Ice.Admin.Facets",
str.str());
815 initData.properties->setProperty(
"IceMX.Metrics.NetworkStats.GroupBy",
"none");
818 return Ice::Application::doMain(argc, argv,
id, i);
824 const std::string configFileName =
"default.cfg";
825 const std::string generatedConfigFileName =
"default.generated.cfg";
826 Ice::StringSeq defaultsPaths;
828 std::filesystem::path absBasePath;
831 absBasePath = defaultsPathVar;
837 defaultsPaths.push_back((absBasePath / configFileName).
string());
838 defaultsPaths.push_back((absBasePath / generatedConfigFileName).
string());
839 return defaultsPaths;
854 for (std::string path : defaultsPath)
860 Ice::PropertyDict defaultCfg = p->getPropertiesForPrefix(
"");
863 for (
auto e : defaultCfg)
865 if (properties->getProperty(e.first).empty())
867 properties->setProperty(e.first, e.second);
871 catch (Ice::FileException& e)
884 const Ice::StringSeq&
887 return ProjectDependendencies;
899 std::string dependenciesConfig =
getIceProperties()->getProperty(
"ArmarX.DependenciesConfig");
901 if (!dependenciesConfig.empty())
903 if (std::filesystem::exists(dependenciesConfig))
906 prop->load(dependenciesConfig);
908 ProjectName = prop->getProperty(
"ArmarX.ProjectName");
909 std::string dependencies = prop->getProperty(
"ArmarX.ProjectDependencies");
925 defineOptionalProperty<std::string>(
926 "Config",
"",
"Comma-separated list of configuration files ");
927 defineOptionalProperty<std::string>(
928 "DependenciesConfig",
929 "./config/dependencies.cfg",
930 "Path to the (usually generated) config file containing all data paths of all dependent "
931 "projects. This property usually does not need to be edited.");
932 defineOptionalProperty<Ice::StringSeq>(
939 "RobotSkillTemplates",
945 "List of ArmarX packages which are accessible by default. Comma separated List. If you "
946 "want to add your own packages and use all default ArmarX packages, use the property "
947 "'AdditionalPackages'.");
948 defineOptionalProperty<Ice::StringSeq>(
949 "AdditionalPackages",
951 "List of additional ArmarX packages which should be in the list of default packages. If "
952 "you have custom packages, which should be found by the gui or other apps, specify them "
953 "here. Comma separated List.");
955 defineOptionalProperty<bool>(
956 "AutodiscoverPackages",
958 "If enabled, will discover all ArmarX packages based on the environment variables. "
959 "Otherwise, the `DefaultPackages` and `AdditionalPackages` properties are used.");
961 defineOptionalProperty<std::string>(
"ApplicationName",
"",
"Application name");
963 defineOptionalProperty<bool>(
"DisableLogging",
965 "Turn logging off in whole application",
968 defineOptionalProperty<MessageTypeT>(
"Verbosity",
970 "Global logging level for whole application",
973 defineOptionalProperty<std::string>(
976 "The logging group is transmitted with every ArmarX log message over Ice in order to group "
977 "the message in the GUI.");
980 defineOptionalProperty<std::string>(
981 "DataPath",
"",
"Semicolon-separated search list for data files");
983 defineOptionalProperty<std::string>(
986 std::string(
"Path for cache files. If relative path AND env. variable ") +
988 " is set, the cache path will be made relative to " +
990 ". Otherwise if relative it will be relative to the default ArmarX config dir (" +
993 defineOptionalProperty<bool>(
"EnableProfiling",
995 "Enable profiling of CPU load produced by this application",
998 defineOptionalProperty<bool>(
999 "RedirectStdout",
true,
"Redirect std::cout and std::cerr to ArmarXLog");
1001 defineOptionalProperty<std::string>(
1004 "Suffix appended to all topic names for outgoing topics. This is mainly used to direct all "
1005 "topics to another name for TopicReplaying purposes.");
1007 defineOptionalProperty<bool>(
1008 "UseTimeServer",
false,
"Enable using a global Timeserver (e.g. from ArmarXSimulator)");
1010 defineOptionalProperty<unsigned int>(
1011 "RemoteHandlesDeletionTimeout",
1013 "The timeout (in ms) before a remote handle deletes the managed object after the use count "
1014 "reached 0. This time can be used by a client to increment the count again (may be "
1015 "required when transmitting remote handles)");
1016 defineOptionalProperty<bool>(
"StartDebuggerOnCrash",
1018 "If this application crashes (segmentation fault) qtcreator will "
1019 "attach to this process and start the debugger.");
1020 defineOptionalProperty<unsigned int>(
1021 "ThreadPoolSize", 1,
"Size of the ArmarX ThreadPool that is always running.");
1022 defineOptionalProperty<std::string>(
1025 "Libraries to load at start up of the application. Must be enabled by the Application with "
1026 "enableLibLoading(). Format: PackageName:LibraryName;... or /absolute/path/to/library;...");
1027 defineOptionalProperty<std::uint64_t>(
1028 "SecondsStartupDelay",
1030 "The startup will be delayed by this number of seconds (useful for debugging)");