Application.cpp
Go to the documentation of this file.
1 /*
2  * This file is part of ArmarX.
3  *
4  * Copyright (C) 2011-2016, High Performance Humanoid Technologies (H2T), Karlsruhe Institute of Technology (KIT), all rights reserved.
5  *
6  * ArmarX is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License version 2 as
8  * published by the Free Software Foundation.
9  *
10  * ArmarX is distributed in the hope that it will be useful, but
11  * WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program. If not, see <http://www.gnu.org/licenses/>.
17  *
18  * @package ArmarXCore::core
19  * @author Nils Adermann (naderman at naderman dot de)
20  * @author Kai Welke (welke at kit dot edu)
21  * @author Jan Issac (jan dot issac at gmail dot com)
22  * @author Mirko Waechter (waechter at kit dot edu)
23  * @author Manfred Kroehnert (manfred dot kroehnert at kit dot edu)
24  * @date 2010
25  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
26  * GNU General Public License
27  */
28 
29 #include "Application.h"
30 
31 #include <stddef.h> // for size_t
32 #include <stdlib.h> // for NULL, getenv, system, exit, etc
33 
34 #include <filesystem>
35 #include <fstream>
36 #include <iostream> // for operator<<, ostream, etc
37 #include <sstream> // for basic_stringbuf<>::int_type, etc
38 #include <thread>
39 #include <thread> // for thread
40 #include <utility> // for pair, move
41 
42 #include <unistd.h> // for getpid
43 
44 #include <Ice/Communicator.h> // for Communicator, etc
45 #include <Ice/Initialize.h> // for InitializationData
46 #include <Ice/LocalException.h> // for FileException, etc
47 #include <Ice/NativePropertiesAdmin.h>
48 #include <Ice/ObjectF.h> // for upCast
49 #include <Ice/Process.h> // for ProcessPtr, Process
50 #include <Ice/Properties.h> // for Properties, PropertiesPtr
51 #include <Ice/Properties.h>
52 #include <Ice/PropertiesAdmin.h> // for PropertyDict
53 #include <Ice/PropertiesAdmin.h>
54 #include <IceUtil/Time.h> // for Time
55 
56 #include <SimoxUtility/algorithm/string/string_tools.h>
57 
58 #include "ArmarXCore/core/IceManager.h" // for IceManager
69 #include "ArmarXCore/core/logging/LogSender.h" // for LogSender
70 #include "ArmarXCore/core/logging/Logging.h" // for ARMARX_INFO_S, etc
71 #include "ArmarXCore/interface/core/Log.h" // for MessageType, etc
72 #include "ArmarXCore/interface/core/Profiler.h"
76 
77 #include "../logging/ArmarXLogBuf.h" // for ArmarXLogBuf
78 #include "ApplicationNetworkStats.h" // for ApplicationNetworkStatsPtr
79 #include "ApplicationOptions.h" // for Options, showHelp, etc
80 #include "ApplicationProcessFacet.h" // for ArmarXManagerPtr, etc
81 #include "properties/IceProperties.h" // for IceProperties
82 
83 #ifndef WIN32
84 #include <signal.h> // for signal, SIGABRT, SIGSEGV, etc
85 
86 #include <cstdio>
87 
88 #include <ArmarXCore/core/util/OnScopeExit.h> // for ARMARX_ON_SCOPE_EXIT
89 
90 #include <execinfo.h>
91 #endif
92 
93 #include <stdlib.h>
94 
95 
96 using namespace armarx;
97 
98 // static members for the one application instance (maybe not required anymore???)
99 std::mutex Application::instanceMutex;
100 ApplicationPtr Application::instance;
101 Ice::StringSeq Application::ProjectDependendencies;
102 std::string Application::ProjectName;
103 const std::string Application::ArmarXUserConfigDirEnvVar = "ARMARX_CONFIG_DIR";
104 std::string errormsg = "Segmentation fault - Backtrace: \n";
105 bool crashed = false;
106 
107 void
109 {
110  if (crashed)
111  {
112  return;
113  }
114  crashed = true;
115 
116  // dont allocate memory in segfault
117  if (sig == SIGSEGV)
118  {
119  std::array<void*, 20> array;
120  size_t size;
121  size = backtrace(array.data(), 20);
122 
123 #pragma GCC diagnostic push
124 #pragma GCC diagnostic ignored "-Wunused-result"
125  write(STDERR_FILENO, errormsg.data(), errormsg.size());
126 #pragma GCC diagnostic pop
127  backtrace_symbols_fd(array.data(), size, STDERR_FILENO);
128  exit(EXIT_FAILURE);
129  }
130  // print out all the frames to stderr
131  try
132  {
133  std::stringstream str;
134 
135  if (sig == SIGABRT)
136  {
137  str << "Error: Abort\nBacktrace:\n";
138  }
139  else
140  {
141  str << "Error: signal " << sig;
142  }
143 
144  ARMARX_FATAL_S << str.str() << "\nBacktrace:\n" << LogSender::CreateBackTrace();
145  if ((sig == SIGSEGV || sig == SIGABRT) && Application::getInstance() &&
146  Application::getInstance()->getProperty<bool>("StartDebuggerOnCrash").getValue())
147  {
148  std::stringstream s;
149  int res;
150  // res = system("which qtcreator");
151  // res = WEXITSTATUS(res);
152  // if (res == EXIT_SUCCESS)
153  {
154  s << "qtcreator -debug " << getpid();
155  ARMARX_IMPORTANT << s.str();
156  res = system(s.str().c_str());
157  res++; // suppress warning
158  }
159  // else
160  // {
161  // ARMARX_INFO_S << "Could not find qtcreator";
162  // }
163  }
164  }
165  catch (...)
166  {
167  // do nothing
168  }
169  exit(EXIT_FAILURE);
170 }
171 
172 void
173 Application::loadLibrariesFromProperties()
174 {
175  auto loadLibString = getProperty<std::string>("LoadLibraries").getValue();
176  if (!loadLibString.empty())
177  {
178  if (!libLoadingEnabled)
179  {
180  throw LocalException("Loading of dynamic libraries is not enabled in this application. "
181  "The Application needs to call enableLibLoading()");
182  }
183  auto entries = armarx::Split(loadLibString, ";", true, true);
184  for (auto& entry : entries)
185  {
186  try
187  {
188  std::filesystem::path path;
189  if (armarx::Contains(entry, "/"))
190  {
191  path = entry;
192  }
193  else
194  {
195  auto elements = Split(entry, ":");
196  ARMARX_CHECK_EQUAL(elements.size(), 2);
197 
198  CMakePackageFinder p(elements.at(0));
199 
200  // case entry ^= `package_name`::`package_name`_`lib_name`
201  {
202  std::string libFileName = "lib" + elements.at(1) + "." +
204  for (auto& libDir : armarx::Split(p.getLibraryPaths(), ";"))
205  {
206  std::filesystem::path testPath(libDir);
207  testPath /= libFileName;
208  if (std::filesystem::exists(testPath))
209  {
210  path = testPath;
211  break;
212  }
213  }
214  }
215  // case entry ^= `package_name`::`lib_name`
216  {
217  // the filename is known to be "lib`package_name`_`lib_name`.so"
218  std::string libFileName = "lib" + elements.at(0) + "_" + elements.at(1) +
219  "." +
221  for (auto& libDir : armarx::Split(p.getLibraryPaths(), ";"))
222  {
223  std::filesystem::path testPath(libDir);
224  testPath /= libFileName;
225  if (std::filesystem::exists(testPath))
226  {
227  path = testPath;
228  break;
229  }
230  }
231  }
232 
233  if (path.empty())
234  {
235  ARMARX_ERROR << "Could find library '" << entry
236  << "' in any of the following paths: " << p.getLibraryPaths();
237  continue;
238  }
239  }
240  ARMARX_CHECK_EXPRESSION(!path.empty());
241  DynamicLibrary lib;
242  lib.setUnloadOnDestruct(false);
243  ARMARX_VERBOSE << "Loading library " << path.string();
244  lib.load(path);
245  }
246  catch (...)
247  {
249  }
250  }
251  }
252 }
253 
254 bool
256 {
257  return forbidThreadCreation;
258 }
259 
260 void
262 {
263  forbidThreadCreation = value;
264  if (forbidThreadCreation)
265  {
266  ARMARX_INFO << "Thread creation with RunningTask and PeriodicTask is now forbidden in this "
267  "process.";
268  }
269  else
270  {
271  ARMARX_INFO << "Thread creation with RunningTask and PeriodicTask is now allowed again in "
272  "this process.";
273  }
274 }
275 
276 void
278 {
279  this->libLoadingEnabled = enable;
280 }
281 
282 void
284 {
285  ARMARX_CHECK_EXPRESSION(commandLineArguments.empty())
286  << "Command line arguments were already set: " << VAROUT(commandLineArguments);
287  commandLineArguments.reserve(argc);
288  for (int i = 0; i < argc; ++i)
289  {
290  commandLineArguments.emplace_back(argv[i]);
291  }
292 }
293 
294 const std::vector<std::string>&
296 {
297  return commandLineArguments;
298 }
299 
300 void
302 {
304  {
305  Application::getInstance()->interruptCallback(sig);
306  }
307 }
308 
309 // main entry point of Ice::Application
311 {
312 }
313 
316 {
317  std::unique_lock lock(instanceMutex);
318 
319  return instance;
320 }
321 
322 void
324 {
325  std::unique_lock lock(instanceMutex);
326 
327  instance = inst;
328 }
329 
330 void
332 {
333  // call base class method
334  PropertyUser::setIceProperties(properties);
335  // set the properties of all components
336  if (armarXManager)
337  {
338  armarXManager->setComponentIceProperties(properties);
339  }
340 }
341 
342 void
343 Application::updateIceProperties(const Ice::PropertyDict& properties)
344 {
345  // call base class method
347  // update the properties of all components
348  if (armarXManager)
349  {
350  armarXManager->updateComponentIceProperties(properties);
351  }
352 }
353 
354 void
355 Application::icePropertiesUpdated(const std::set<std::string>& changedProperties)
356 {
357  if (armarXManager)
358  {
359  if (changedProperties.count("DisableLogging") &&
360  getProperty<bool>("DisableLogging").isSet())
361  {
362  // ARMARX_INFO << "Logging disabled: " << getProperty<bool>("DisableLogging").getValue();
363  armarXManager->enableLogging(!getProperty<bool>("DisableLogging").getValue());
364  }
365 
366  if (changedProperties.count("EnableProfiling"))
367  {
368  armarXManager->enableProfiling(getProperty<bool>("EnableProfiling").getValue());
369  }
370  if (changedProperties.count("Verbosity"))
371  {
372  armarXManager->setGlobalMinimumLoggingLevel(
373  getProperty<MessageTypeT>("Verbosity").getValue());
374  }
375  }
376 }
377 
378 const ThreadPoolPtr&
380 {
381  return threadPool;
382 }
383 
384 int
385 Application::run(int argc, char* argv[])
386 {
387 #ifndef WIN32
388  // register signal handler
389  signal(SIGILL, Application::HandlerFault);
390  signal(SIGSEGV, Application::HandlerFault);
391  signal(SIGABRT, Application::HandlerFault);
392 
393  signal(SIGHUP, Application::HandlerInterrupt);
394  signal(SIGINT, Application::HandlerInterrupt);
395  signal(SIGTERM, Application::HandlerInterrupt);
396 #endif
397 
398  ARMARX_TRACE;
399 
400  // parse options and merge these with properties passed via Ice.Config
401  Ice::PropertiesPtr properties = parseOptionsMergeProperties(argc, argv);
402 
403  // parse help options
404  ARMARX_TRACE;
405 
407  ApplicationOptions::parseHelpOptions(properties, argc, argv);
408 
409  if (options.error)
410  {
411  // error in options
412  return EXIT_FAILURE;
413  }
414  else if (options.showHelp) // display help
415  {
416  showHelp(options);
417  return EXIT_SUCCESS;
418  }
419  else if (options.showVersion)
420  {
421  std::cout << "Version: " << GetVersion() << std::endl;
422  return EXIT_SUCCESS;
423  }
424 
425  ARMARX_TRACE;
426 
427  setIceProperties(properties);
428 
429  // Override environment variables if set.
430  {
431  ARMARX_TRACE;
432 
433  const Ice::PropertyDict& env_props =
434  getPropertyDefinitions()->getProperties()->getPropertiesForPrefix("env");
435 
436  for (const auto& [name, value] : env_props)
437  {
438  const std::string short_name = name.substr(4); // Cut "env.".
439  ARMARX_INFO << "Setting environment variable '" << short_name << "' to value '" << value
440  << "'.";
441  setenv(short_name.c_str(), value.c_str(), 1);
442  }
443  }
444 
445  if (getProperty<std::uint64_t>("SecondsStartupDelay").isSet())
446  {
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});
450  }
451  else
452  {
453  ARMARX_DEBUG << "startup delay is not set";
454  }
455 
456  // extract application name
457  if (getProperty<std::string>("ApplicationName").isSet())
458  {
459  applicationName = getProperty<std::string>("ApplicationName").getValue();
460  }
461  LogSender::SetLoggingGroup(getProperty<std::string>("LoggingGroup").getValue());
462 
463  // Redirect std::cout and std::cerr
464  const bool redirectStdout = getProperty<bool>("RedirectStdout").getValue();
465  ArmarXLogBuf buf("std::cout", MessageTypeT::INFO, false);
466  ArmarXLogBuf errbuf("std::cerr", MessageTypeT::WARN, true);
467  std::streambuf* cout_sbuf = nullptr;
468  std::streambuf* cerr_sbuf = nullptr;
469  if (redirectStdout)
470  {
471  cout_sbuf = std::cout.rdbuf();
472  cerr_sbuf = std::cerr.rdbuf();
473  std::cout.rdbuf(&buf);
474  std::cerr.rdbuf(&errbuf);
475  }
477  {
478  //reset std::cout and std::cerr to original stream
479  if (redirectStdout)
480  {
481  std::cout.rdbuf(cout_sbuf);
482  std::cerr.rdbuf(cerr_sbuf);
483  }
484  };
485 
486  int result = EXIT_SUCCESS;
487 
488  threadPool.reset(new ThreadPool(getProperty<unsigned int>("ThreadPoolSize").getValue()));
489  // create the ArmarXManager and set its properties
490  try
491  {
492  armarXManager = new ArmarXManager(applicationName, communicator());
493  }
494  catch (Ice::ConnectFailedException&)
495  {
496  return EXIT_FAILURE;
497  }
498 
499  armarXManager->setDataPaths(getProperty<std::string>("DataPath").getValue());
500 
501  // set the properties again, since the armarx manager is now available
502  setIceProperties(properties);
503 
505 
506 #ifdef WIN32
507  // register interrupt handler
508  callbackOnInterrupt();
509 #endif
510 
511 
512  if (applicationNetworkStats)
513  {
514  ProfilerListenerPrx profilerTopic =
515  armarXManager->getIceManager()->getTopic<ProfilerListenerPrx>(
516  armarx::Profiler::PROFILER_TOPIC_NAME);
517  applicationNetworkStats->start(profilerTopic, applicationName);
518  }
519 
520  loadLibrariesFromProperties();
521 
522  try
523  {
524  // calls virtual setup in order to allow subclass to add components to the application
525  setup(armarXManager, properties);
526  }
527  catch (...)
528  {
530  armarXManager->shutdown();
531  }
532 
533  // calls exec implementation
534  result = exec(armarXManager);
536 
537  if (shutdownThread)
538  {
539  ARMARX_VERBOSE << "joining shutdown thread!";
540  shutdownThread->join();
541  }
542  ARMARX_VERBOSE << "Application run finished";
543  return result;
544 }
545 
546 /*
547  * the default exec implementation. Exec is always blocking and waits for
548  * shutdown
549  */
550 int
551 Application::exec(const ArmarXManagerPtr& armarXManager)
552 {
553  armarXManager->waitForShutdown();
554 
555  return 0;
556 }
557 
560 {
561  // parse options and merge these with properties passed via Ice.Config
562  // Here, use armarx::IceProperties instead of Ice::Properties
563  // to support property inheritance.
565  ApplicationOptions::mergeProperties(communicator()->getProperties()->clone(), argc, argv));
566 
567  // Load config files passed via ArmarX.Config
568  std::string configFiles = communicator()->getProperties()->getProperty("ArmarX.Config");
569 
570  PropertyDefinition<std::string>::removeQuotes(configFiles, configFiles);
571 
572  // if not set, size is zero
573  // if set with out value (== "1" as TRUE), size is one
574  if (!configFiles.empty() && configFiles.compare("1") != 0)
575  {
576  std::vector<std::string> configFileList = simox::alg::split(configFiles, ",");
577 
578  for (std::string configFile : configFileList)
579  {
580  if (!configFile.empty())
581  {
582  properties->load(configFile);
583  }
584  }
585  }
586  return properties;
587 }
588 
589 void
591 {
592  // perform dummy setup to register ManagedIceObjects
593  ArmarXDummyManagerPtr dummyManager = new ArmarXDummyManager();
594 
595  LogSender::SetLoggingActivated(false, false);
596 
597  setup(dummyManager, getIceProperties());
598 
599  if (options.outfile.empty())
600  {
601  ApplicationOptions::showHelp(this, dummyManager, options, nullptr);
602  }
603  else
604  {
605  std::ofstream out(options.outfile);
606  if (out.is_open())
607  {
608  ApplicationOptions::showHelp(this, dummyManager, options, nullptr, out);
609  }
610  else
611  {
613  ApplicationOptions::showHelp(this, dummyManager, options, nullptr);
614 
615  std::cout << "Could not write to file " << options.outfile << std::endl;
616  }
617  }
618 }
619 
620 std::string
622 {
623  return "ArmarX";
624 }
625 
626 void
627 Application::setName(const std::string& name)
628 {
629  this->applicationName = name;
630 }
631 
632 std::string
634 {
635  return applicationName;
636 }
637 
638 void
640 {
641  ARMARX_INFO_S << "Interrupt received: " << signal;
642  if (applicationNetworkStats)
643  {
644  applicationNetworkStats->stopTask();
645  }
646  if (!shutdownThread)
647  shutdownThread.reset(new std::thread{[this]
648  {
649  if (armarXManager)
650  this->armarXManager->shutdown();
651  }});
652 }
653 
654 void
656 {
657  IceUtil::Time start = IceUtil::Time::now();
658  ARMARX_INFO_S << "Deps: " << dependencies;
659  dependencies = simox::alg::replace_all(dependencies, "\"", "");
660  Ice::StringSeq resultList = simox::alg::split(dependencies, "/");
661 
662  for (size_t i = 0; i < resultList.size(); i++)
663  {
664  CMakePackageFinder pack(resultList[i]);
665 
666  if (pack.packageFound())
667  {
669  }
670  }
671 
672  ARMARX_INFO_S << "loading took " << (IceUtil::Time::now() - start).toMilliSeconds();
673 }
674 
675 bool
677 {
678  const std::string autodiscoverPropertyName = "AutodiscoverPackages";
679 
680  if (hasProperty(autodiscoverPropertyName))
681  {
682  ARMARX_VERBOSE << "`" << autodiscoverPropertyName << "` property available.";
683  return getProperty<bool>(autodiscoverPropertyName).getValue();
684  }
685 
686  return true; // Property not available. Default: use autodiscovery
687 }
688 
689 std::vector<std::string>
691 {
693  {
694  ARMARX_VERBOSE << "ArmarX package autodiscovery activated.";
696  }
697 
698  ARMARX_VERBOSE << "ArmarX package autodiscovery not activated.";
699 
700  // legacy: use only those packages listed in the config file (e.g. default.cfg)
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());
704 
705  return result;
706 }
707 
708 std::string
710 {
711  char* env_armarx_workspace = getenv("ARMARX_WORKSPACE");
712  char* env_armarx_default_config_dir_name = getenv("ARMARX_CONFIG_DIR_NAME");
713 
714  std::filesystem::path armarx_workspace;
715  std::filesystem::path armarx_config_dir;
716 
717  if (env_armarx_workspace)
718  {
719  armarx_workspace = std::filesystem::path(env_armarx_workspace);
720  }
721  else
722  {
723  char* home = getenv("HOME");
724 
725  if (home)
726  {
727  armarx_workspace = std::filesystem::path(home);
728  }
729  else
730  {
731  armarx_workspace = "~/";
732  }
733  }
734 
735  if (env_armarx_default_config_dir_name)
736  {
737  armarx_config_dir = std::filesystem::path(env_armarx_default_config_dir_name);
738  }
739  else
740  {
741  if (env_armarx_workspace)
742  {
743  armarx_config_dir = "armarx_config";
744  }
745  // Legacy mode.
746  else
747  {
748  armarx_config_dir = ".armarx";
749  }
750  }
751 
752  if (envVarExpanded)
753  {
754  return (armarx_workspace / armarx_config_dir).string();
755  }
756  else
757  {
758  if (env_armarx_workspace)
759  {
760  return "${ARMARX_WORKSPACE}/" + armarx_config_dir.string();
761  }
762  // Legacy mode.
763  else
764  {
765  return "${HOME}/" + armarx_config_dir.string();
766  }
767  }
768 }
769 
772 {
773  return armarXManager;
774 }
775 
776 int
777 Application::doMain(int argc, char* argv[], const Ice::InitializationData& initData, Ice::Int i)
778 {
779  // create copy of initData so the stats object can be added
780  // using const_cast leads to unexpected errors
781  Ice::InitializationData id(initData);
782  loadDefaultConfig(argc, argv, id);
783 
784  // enable the PropertyAdmin ObjectAdapter
785  // https://doc.zeroc.com/display/Ice35/The+Administrative+Object+Adapter
786  if (id.properties->getProperty("Ice.Admin.InstanceName").empty())
787  {
788  id.properties->setProperty("Ice.Admin.InstanceName", this->getName());
789  }
790  if (id.properties->getProperty("Ice.Admin.Endpoints").empty())
791  {
792  id.properties->setProperty("Ice.Admin.Endpoints", "tcp -h 127.0.0.1");
793  }
794 
795  if (initData.properties->getProperty("ArmarX.NetworkStats") == "1")
796  {
797  applicationNetworkStats = new ApplicationNetworkStats();
798  id.observer = applicationNetworkStats;
799 
800  const auto oldFacets = initData.properties->getPropertyAsList("Ice.Admin.Facets");
801  if (oldFacets.empty())
802  {
803  initData.properties->setProperty("Ice.Admin.Facets", "Metrics");
804  }
805  else if (std::find(oldFacets.begin(), oldFacets.end(), "Metrics") == oldFacets.end())
806  {
807  std::stringstream str;
808  str << "Metrics";
809  for (const auto& facet : oldFacets)
810  {
811  str << "," << facet;
812  }
813  initData.properties->setProperty("Ice.Admin.Facets", str.str());
814  }
815  initData.properties->setProperty("IceMX.Metrics.NetworkStats.GroupBy", "none");
816  }
817 
818  return Ice::Application::doMain(argc, argv, id, i);
819 }
820 
821 Ice::StringSeq
823 {
824  const std::string configFileName = "default.cfg";
825  const std::string generatedConfigFileName = "default.generated.cfg";
826  Ice::StringSeq defaultsPaths;
827  char* defaultsPathVar = getenv(ArmarXUserConfigDirEnvVar.c_str());
828  std::filesystem::path absBasePath;
829  if (defaultsPathVar)
830  {
831  absBasePath = defaultsPathVar;
832  }
833  else
834  {
835  absBasePath = std::filesystem::path(GetArmarXConfigDefaultPath());
836  }
837  defaultsPaths.push_back((absBasePath / configFileName).string());
838  defaultsPaths.push_back((absBasePath / generatedConfigFileName).string());
839  return defaultsPaths;
840 }
841 
842 void
843 Application::loadDefaultConfig(int argc, char* argv[], const Ice::InitializationData& initData)
844 {
845  LoadDefaultConfig(initData.properties);
846 }
847 
848 void
850 {
851  const Ice::StringSeq defaultsPath = GetDefaultsPaths();
853 
854  for (std::string path : defaultsPath)
855  {
856  try
857  {
858  p->load(path);
859 
860  Ice::PropertyDict defaultCfg = p->getPropertiesForPrefix("");
861 
862  // copy default config into current config (if values are not already there)
863  for (auto e : defaultCfg)
864  {
865  if (properties->getProperty(e.first).empty())
866  {
867  properties->setProperty(e.first, e.second);
868  }
869  }
870  }
871  catch (Ice::FileException& e)
872  {
873  ARMARX_WARNING_S << "Loading default config failed: " << e.what();
874  }
875  }
876 }
877 
878 const std::string&
880 {
881  return ProjectName;
882 }
883 
884 const Ice::StringSeq&
886 {
887  return ProjectDependendencies;
888 }
889 
892 {
894 }
895 
896 void
898 {
899  std::string dependenciesConfig = getIceProperties()->getProperty("ArmarX.DependenciesConfig");
900 
901  if (!dependenciesConfig.empty())
902  {
903  if (std::filesystem::exists(dependenciesConfig))
904  {
906  prop->load(dependenciesConfig);
907  ArmarXDataPath::addDataPaths(prop->getProperty("ArmarX.ProjectDatapath"));
908  ProjectName = prop->getProperty("ArmarX.ProjectName");
909  std::string dependencies = prop->getProperty("ArmarX.ProjectDependencies");
910  ProjectDependendencies = simox::alg::split(dependencies, ";");
911  }
912  else
913  {
914  /*
915  ARMARX_WARNING_S << "The given project datapath config file '" << datapathConfig << "', that this app depends on, could not be found. \
916  Relative paths to subprojects are not available. Set the property ArmarX.ProjectDatapath to empty, if you do not need the paths.";
917  */
918  }
919  }
920 }
921 
924 {
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>(
933  "DefaultPackages",
934  {"ArmarXCore",
935  "ArmarXGui",
936  "MemoryX",
937  "RobotAPI",
938  "RobotComponents",
939  "RobotSkillTemplates",
940  "ArmarXSimulation",
941  "VisionX",
942  "SpeechX",
943  "Armar3",
944  "Spoac"},
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",
950  {},
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.");
954 
955  defineOptionalProperty<bool>(
956  "AutodiscoverPackages",
957  true,
958  "If enabled, will discover all ArmarX packages based on the environment variables. "
959  "Otherwise, the `DefaultPackages` and `AdditionalPackages` properties are used.");
960 
961  defineOptionalProperty<std::string>("ApplicationName", "", "Application name");
962 
963  defineOptionalProperty<bool>("DisableLogging",
964  false,
965  "Turn logging off in whole application",
967 
968  defineOptionalProperty<MessageTypeT>("Verbosity",
970  "Global logging level for whole application",
972 
973  defineOptionalProperty<std::string>(
974  "LoggingGroup",
975  "",
976  "The logging group is transmitted with every ArmarX log message over Ice in order to group "
977  "the message in the GUI.");
978 
979 
980  defineOptionalProperty<std::string>(
981  "DataPath", "", "Semicolon-separated search list for data files");
982 
983  defineOptionalProperty<std::string>(
984  "CachePath",
985  "mongo/.cache",
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 (" +
992 
993  defineOptionalProperty<bool>("EnableProfiling",
994  false,
995  "Enable profiling of CPU load produced by this application",
997 
998  defineOptionalProperty<bool>(
999  "RedirectStdout", true, "Redirect std::cout and std::cerr to ArmarXLog");
1000 
1001  defineOptionalProperty<std::string>(
1002  "TopicSuffix",
1003  "",
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.");
1006 
1007  defineOptionalProperty<bool>(
1008  "UseTimeServer", false, "Enable using a global Timeserver (e.g. from ArmarXSimulator)");
1009 
1010  defineOptionalProperty<unsigned int>(
1011  "RemoteHandlesDeletionTimeout",
1012  3000,
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",
1017  false,
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>(
1023  "LoadLibraries",
1024  "",
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",
1029  0,
1030  "The startup will be delayed by this number of seconds (useful for debugging)");
1031 }
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::Application::run
int run(int argc, char *argv[]) override
Ice::Application replacement for the main function.
Definition: Application.cpp:385
armarx::Application::GetProjectName
static const std::string & GetProjectName()
Definition: Application.cpp:879
armarx::Application::Application
Application()
Application initalizes the Ice::Application base class.
Definition: Application.cpp:310
str
std::string str(const T &t)
Definition: UserAssistedSegmenterGuiWidgetController.cpp:43
armarx::Application::getCommandLineArguments
const std::vector< std::string > & getCommandLineArguments() const
Definition: Application.cpp:295
armarx::ApplicationOptions::mergeProperties
Ice::PropertiesPtr mergeProperties(Ice::PropertiesPtr properties, int argc, char *argv[])
Merge command line options into properties.
Definition: ApplicationOptions.cpp:63
errormsg
std::string errormsg
Definition: Application.cpp:104
armarx::Application::exec
virtual int exec(const ArmarXManagerPtr &armarXManager)
Exec method is the main process of the application.
Definition: Application.cpp:551
armarx::MessageTypeT::WARN
@ WARN
ARMARX_IMPORTANT
#define ARMARX_IMPORTANT
Definition: Logging.h:190
armarx::CMakePackageFinder::packageFound
bool packageFound() const
Returns whether or not this package was found with cmake.
Definition: CMakePackageFinder.cpp:511
armarx::Application::setup
virtual void setup(const ManagedIceObjectRegistryInterfacePtr &registry, Ice::PropertiesPtr properties)=0
Setup method to be implemented by user applications.
OnScopeExit.h
armarx::Application::storeCommandLineArguments
void storeCommandLineArguments(int argc, char *argv[])
Definition: Application.cpp:283
ArmarXManager.h
armarx::MessageTypeT::INFO
@ INFO
armarx::Application::parseOptionsMergeProperties
Ice::PropertiesPtr parseOptionsMergeProperties(int argc, char *argv[])
Parse options given on the commandline and merge them into the regular properties.
Definition: Application.cpp:559
DynamicLibrary.h
armarx::CMakePackageFinder::FindAllArmarXSourcePackages
static std::vector< std::string > FindAllArmarXSourcePackages()
Definition: CMakePackageFinder.cpp:461
armarx::Contains
bool Contains(const ContainerType &container, const ElementType &searchElement)
Definition: algorithm.h:330
armarx::ThreadPool
The ThreadPool class.
Definition: ThreadPool.h:45
armarx::DynamicLibrary::load
void load(std::filesystem::path libPath)
Loads a shared library from the specified path.
Definition: DynamicLibrary.cpp:54
armarx::Application::HandlerFault
static void HandlerFault(int sig)
handlerFault handles signals sendt to the application such as SIGSEGF or SIGABRT (Linux)
Definition: Application.cpp:108
armarx::Application::createPropertyDefinitions
PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: Application.cpp:891
armarx::Split
std::vector< std::string > Split(const std::string &source, const std::string &splitBy, bool trimElements=false, bool removeEmptyElements=false)
Definition: StringHelperTemplates.h:36
armarx::LogSender::SetSendLoggingActivated
static void SetSendLoggingActivated(bool activated=true)
Definition: LogSender.cpp:740
armarx::Application::GetDefaultsPaths
static Ice::StringSeq GetDefaultsPaths()
Definition: Application.cpp:822
armarx::CMakePackageFinder
The CMakePackageFinder class provides an interface to the CMake Package finder capabilities.
Definition: CMakePackageFinder.h:52
armarx::Application::loadDefaultConfig
void loadDefaultConfig(int argc, char *argv[], const Ice::InitializationData &initData)
Definition: Application.cpp:843
armarx::ApplicationOptions::eHelpBrief
@ eHelpBrief
Definition: ApplicationOptions.h:64
armarx::ApplicationOptions::parseHelpOptions
Options parseHelpOptions(Ice::PropertiesPtr properties, int argc, char *argv[])
Parse the help options.
Definition: ApplicationOptions.cpp:73
armarx::LogSender::SetLoggingGroup
static void SetLoggingGroup(const std::string &loggingGroup)
Definition: LogSender.cpp:153
ARMARX_ON_SCOPE_EXIT
#define ARMARX_ON_SCOPE_EXIT
Executes given code when the enclosing scope is left.
Definition: OnScopeExit.h:120
armarx::Application::GetArmarXConfigDefaultPath
static std::string GetArmarXConfigDefaultPath(bool envVarExpanded=true)
Definition: Application.cpp:709
armarx::ApplicationOptions::Options::showHelp
bool showHelp
Definition: ApplicationOptions.h:78
armarx::DynamicLibrary::GetSharedLibraryFileExtension
static std::string GetSharedLibraryFileExtension()
Definition: DynamicLibrary.cpp:164
armarx::PropertyUser::hasProperty
bool hasProperty(const std::string &name)
Definition: PropertyUser.cpp:234
ApplicationOptions.h
armarx::Application::getArmarXManager
ArmarXManagerPtr getArmarXManager()
Definition: Application.cpp:771
armarx::ThreadPoolPtr
std::shared_ptr< ThreadPool > ThreadPoolPtr
Definition: Application.h:76
armarx::Application::HandlerInterrupt
static void HandlerInterrupt(int sig)
handlerInterrupt handles interrupt signals sent to the application (Linux)
Definition: Application.cpp:301
IceInternal::Handle<::Ice::Properties >
armarx::ApplicationNetworkStats
The ApplicationNetworkStats class implements the Ice::Instrumentation::CommunicatorObserver interface...
Definition: ApplicationNetworkStats.h:51
armarx::Application::icePropertiesUpdated
void icePropertiesUpdated(const std::set< std::string > &changedProperties) override
This method is called when new Properties are set via setIceProperties().
Definition: Application.cpp:355
ArmarXDummyManager.h
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
cxxopts::value
std::shared_ptr< Value > value()
Definition: cxxopts.hpp:855
armarx::Application::getDomainName
virtual std::string getDomainName()
Retrieve the domain name used for property parsing.
Definition: Application.cpp:621
Property.h
armarx::DynamicLibrary::setUnloadOnDestruct
void setUnloadOnDestruct(bool unload)
Definition: DynamicLibrary.cpp:147
armarx::Application::isPackageAutoDiscoveryEnabled
bool isPackageAutoDiscoveryEnabled()
Definition: Application.cpp:676
armarx::Application::setInstance
static void setInstance(ApplicationPtr const &inst)
Definition: Application.cpp:323
ManagedIceObject.h
ThreadPool.h
PropertyUser.h
armarx::ApplicationOptions::Options::format
OptionsFormat format
Definition: ApplicationOptions.h:82
ARMARX_DEBUG
#define ARMARX_DEBUG
Definition: Logging.h:184
armarx::ApplicationOptions::Options::showVersion
bool showVersion
Definition: ApplicationOptions.h:79
armarx::PropertyUser::updateIceProperties
virtual void updateIceProperties(const std::map< std::string, std::string > &changes)
Definition: PropertyUser.cpp:112
armarx::Application::getForbidThreadCreation
bool getForbidThreadCreation() const
Definition: Application.cpp:255
armarx::Application::registerDataPathsFromDependencies
void registerDataPathsFromDependencies(std::string dependencies)
Definition: Application.cpp:655
armarx::ApplicationPropertyDefinitions::ApplicationPropertyDefinitions
ApplicationPropertyDefinitions(std::string prefix)
Definition: Application.cpp:922
armarx::ApplicationOptions::Options
Stucture containing the parsed options of the application.
Definition: ApplicationOptions.h:76
IceManager.h
armarx::LogSender::CreateBackTrace
static std::string CreateBackTrace(int linesToSkip=1)
Definition: LogSender.cpp:516
armarx::IceProperties::create
static Ice::PropertiesPtr create(const Ice::PropertiesPtr &iceProperties=nullptr)
Definition: IceProperties.cpp:35
ApplicationNetworkStats.h
armarx::Application::getThreadPool
const ThreadPoolPtr & getThreadPool() const
Definition: Application.cpp:379
ARMARX_ERROR
#define ARMARX_ERROR
Definition: Logging.h:196
armarx::PropertyDefinitionContainer
PropertyDefinitionContainer.
Definition: PropertyDefinitionContainer.h:53
armarx::Application::getInstance
static ApplicationPtr getInstance()
Retrieve shared pointer to the application object.
Definition: Application.cpp:315
armarx::Application
Baseclass for all ArmarX applications.
Definition: Application.h:193
armarx::ApplicationPropertyDefinitions
Application property definition container.
Definition: Application.h:101
ARMARX_WARNING_S
#define ARMARX_WARNING_S
Definition: Logging.h:213
armarx::armem::Time
armarx::core::time::DateTime Time
Definition: forward_declarations.h:13
armarx::Application::getName
std::string getName() const
Retrieve name of the application.
Definition: Application.cpp:633
ExpressionException.h
CMakePackageFinder.h
Ice
Definition: DBTypes.cpp:63
armarx::Application::setIceProperties
void setIceProperties(Ice::PropertiesPtr properties) override
Overrides PropertyUser::setIceProperties() which is called internally.
Definition: Application.cpp:331
ARMARX_CHECK_EXPRESSION
#define ARMARX_CHECK_EXPRESSION(expression)
This macro evaluates the expression and if it turns out to be false it will throw an ExpressionExcept...
Definition: ExpressionException.h:73
armarx::Application::LoadDefaultConfig
static void LoadDefaultConfig(Ice::PropertiesPtr properties)
Definition: Application.cpp:849
armarx::LogSender::SetLoggingActivated
static void SetLoggingActivated(bool activated=true, bool showMessage=true)
setLoggingActivated() is used to activate or disable the logging facilities in the whole application
Definition: LogSender.cpp:725
PropertyDefinition.hpp
armarx::ApplicationOptions::Options::error
bool error
Definition: ApplicationOptions.h:80
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::ArmarXDummyManager
Handles help and documentation generation but does not provide Ice functionality.
Definition: ArmarXDummyManager.h:52
armarx::ArmarXDataPath::addDataPaths
static void addDataPaths(const std::string &dataPathList)
Definition: ArmarXDataPath.cpp:554
armarx::PropertyDefinition::removeQuotes
bool removeQuotes() const
Definition: PropertyDefinition.hpp:253
VAROUT
#define VAROUT(x)
Definition: StringHelpers.h:198
IceUtil::Handle
Definition: forward_declarations.h:30
armarx::CMakePackageFinder::getIncludePaths
std::string getIncludePaths() const
Returns the include paths separated by semi-colons.
Definition: CMakePackageFinder.h:114
armarx::Application::getArmarXPackageNames
std::vector< std::string > getArmarXPackageNames()
getDefaultPackageNames returns the value of the ArmarX.DefaultPackages property It splits the string ...
Definition: Application.cpp:690
LogSender.h
crashed
bool crashed
Definition: Application.cpp:105
armarx::PropertyUser::getPropertyDefinitions
PropertyDefinitionsPtr getPropertyDefinitions()
Returns the component's property definition container.
Definition: PropertyUser.cpp:60
armarx::VariantType::Int
const VariantTypeId Int
Definition: Variant.h:917
ARMARX_FATAL_S
#define ARMARX_FATAL_S
Definition: Logging.h:219
armarx::Application::GetVersion
static std::string GetVersion()
Definition: Application.h:235
armarx::aron::write
requires data::isWriter< WriterT > void write(WriterT &aron_w, const Eigen::Matrix< EigenT, rows, cols, options > &input, typename WriterT::ReturnType &ret, const armarx::aron::Path &aron_p=armarx::aron::Path())
Definition: eigen.h:138
armarx::Application::setForbidThreadCreation
void setForbidThreadCreation(bool value)
Definition: Application.cpp:261
ARMARX_INFO_S
#define ARMARX_INFO_S
Definition: Logging.h:202
armarx::PropertyUser::getIceProperties
Ice::PropertiesPtr getIceProperties() const
Returns the set of Ice properties.
Definition: PropertyUser.cpp:221
armarx::DynamicLibrary
The DynamicLibrary class provides a mechanism to load libraries at runtime.
Definition: DynamicLibrary.h:49
armarx::Application::enableLibLoading
void enableLibLoading(bool enable=true)
Definition: Application.cpp:277
Logging.h
armarx::ApplicationOptions::showHelp
void showHelp(ApplicationPtr application, ArmarXDummyManagerPtr dummyManager, Options options, Ice::PropertiesPtr properties, std::ostream &out=std::cout)
Prints help according to the format selection in options.
Definition: ApplicationOptions.cpp:187
armarx::ApplicationOptions::Options::outfile
std::string outfile
Definition: ApplicationOptions.h:81
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
armarx::handleExceptions
void handleExceptions()
Definition: Exception.cpp:157
ApplicationProcessFacet.h
armarx::PropertyDefinitionsPtr
IceUtil::Handle< class PropertyDefinitionContainer > PropertyDefinitionsPtr
PropertyDefinitions smart pointer type.
Definition: forward_declarations.h:35
IceProperties.h
armarx::ArmarXLogBuf
Definition: ArmarXLogBuf.h:33
armarx::Application::interruptCallback
void interruptCallback(int signal) override
Cleans up connections with IceStorm before terminating the app.
Definition: Application.cpp:639
ArmarXDataPath.h
armarx::PropertyDefinitionBase::eModifiable
@ eModifiable
Definition: PropertyDefinitionInterface.h:57
armarx::Application::ArmarXUserConfigDirEnvVar
static const std::string ArmarXUserConfigDirEnvVar
Definition: Application.h:297
armarx::Application::setName
void setName(const std::string &name)
Set name of the application.
Definition: Application.cpp:627
armarx::PropertyUser::setIceProperties
virtual void setIceProperties(Ice::PropertiesPtr properties)
Sets the Ice properties.
Definition: PropertyUser.cpp:87
armarx::Application::loadDependentProjectDatapaths
void loadDependentProjectDatapaths()
Definition: Application.cpp:897
armarx::ctrlutil::s
double s(double t, double s0, double v0, double a0, double j)
Definition: CtrlUtil.h:33
armarx::ArmarXManager
Main class of an ArmarX process.
Definition: ArmarXManager.h:97
armarx::Application::updateIceProperties
void updateIceProperties(const Ice::PropertyDict &properties) override
Definition: Application.cpp:343
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
armarx::Application::GetProjectDependencies
static const Ice::StringSeq & GetProjectDependencies()
Definition: Application.cpp:885
PropertyDefinition.h
Application.h
armarx::split
std::vector< std::string > split(const std::string &source, const std::string &splitBy, bool trimElements=false, bool removeEmptyElements=false)
Definition: StringHelpers.cpp:38
armarx::Application::doMain
int doMain(int argc, char *argv[], const Ice::InitializationData &initData, Ice::Int i) override
Ice::Application::doMain() is called by Ice::Application::main() and does setup of Ice::Communicator ...
Definition: Application.cpp:777
armarx::Application::showHelp
void showHelp(ApplicationOptions::Options &options)
Print help onto the screen or into a file.
Definition: Application.cpp:590