https://mooseframework.inl.gov
Capabilities.C
Go to the documentation of this file.
1 //* This file is part of the MOOSE framework
2 //* https://mooseframework.inl.gov
3 //*
4 //* All rights reserved, see COPYRIGHT for full restrictions
5 //* https://github.com/idaholab/moose/blob/master/COPYRIGHT
6 //*
7 //* Licensed under LGPL 2.1, please see LICENSE for details
8 //* https://www.gnu.org/licenses/lgpl-2.1.html
9 
10 // Needs to be included before Capabilities.h, which
11 // includes nlohmann/json_fwd.h
12 #include "nlohmann/json.h"
13 
14 #include "Capabilities.h"
15 #include "CapabilityException.h"
16 #include "MooseUtils.h"
17 #include "Conversion.h"
18 
19 #include <vector>
20 
21 #include "libmesh/libmesh_config.h"
22 #include "petscconf.h"
23 #include "petscpkg_version.h"
24 
25 #ifdef MOOSE_LIBTORCH_ENABLED
26 #include <torch/version.h>
27 #endif
28 
29 #if __has_include(<torch/xpu.h>)
30 #include <torch/xpu.h>
31 #define MOOSE_HAVE_XPU 1
32 #endif
33 
34 #ifdef MOOSE_MFEM_ENABLED
35 #include "mfem/config/config.hpp"
36 #endif
37 
38 #ifdef LIBMESH_HAVE_NETGEN
39 #include "netgen/netgen_version.hpp"
40 #endif
41 
42 namespace Moose::internal
43 {
44 
46 {
47  // Register the moose capabilities, just once
49 }
50 
53 {
54  // We need a naked new here (_not_ a smart pointer or object instance) due to what seems like a
55  // bug in clang's static object destruction when using dynamic library loading.
56  static Capabilities * capability_registry = nullptr;
57  if (!capability_registry)
58  capability_registry = new Capabilities();
59  return *capability_registry;
60 }
61 
62 std::string
64 {
65  nlohmann::json root;
66  for (const auto & [name, capability] : _registry)
67  {
68  auto & entry = root[name];
69  std::visit([&entry](const auto & v) { entry["value"] = v; }, capability.getValue());
70  entry["doc"] = capability.getDoc();
71  if (const auto enumeration_ptr = capability.queryEnumeration())
72  entry["enumeration"] = *enumeration_ptr;
73  if (!capability.hasBoolValue())
74  entry["explicit"] = capability.getExplicit();
75  }
76  return root.dump(2);
77 }
78 
79 void
80 Capabilities::augment(const nlohmann::json & input, const Capabilities::AugmentPassKey)
81 {
82  for (const auto & [name_json, entry] : input.items())
83  {
84  const std::string name = name_json;
85 
86  const auto error = [&name](const std::string & message)
87  { throw CapabilityException("Capabilities::augment: Capability '" + name + "' " + message); };
88 
89  std::string doc;
90  if (const auto it = entry.find("doc"); it != entry.end())
91  doc = *it;
92  else
93  error("missing 'doc' entry");
94 
96  if (const auto it = entry.find("value"); it != entry.end())
97  {
98  if (it->is_boolean())
99  value = it->get<bool>();
100  else if (it->is_number_integer())
101  value = it->get<int>();
102  else
103  value = it->get<std::string>();
104  }
105  else
106  error("missing 'value' entry");
107 
108  auto & capability = add(name, value, doc);
109 
110  if (const auto it = entry.find("explicit"); it != entry.end() && it->get<bool>())
111  capability.setExplicit();
112 
113  if (const auto it = entry.find("enumeration"); it != entry.end())
114  capability.setEnumeration(it->get<std::set<std::string>>());
115  }
116 }
117 
118 bool
119 Capabilities::isInstallationType(const std::string & installation_type) const
120 {
121  const auto & capability = get("installation_type");
122  mooseAssert(capability.getEnumeration().count(installation_type), "Not a valid enumeration");
123  return capability.getStringValue() == installation_type;
124 }
125 
126 void
128 {
129  // helper lambdas for explicitly adding typed capabilities
130  const auto add_bool = [&](const std::string_view capability,
131  const bool value,
132  const std::string_view doc) -> Capability &
133  { return add(capability, value, doc); };
134  const auto add_int = [&](const std::string_view capability,
135  const int value,
136  const std::string_view doc) -> Capability &
137  { return add(capability, value, doc); };
138  const auto add_string = [&](const std::string_view capability,
139  const std::string_view value,
140  const std::string_view doc) -> Capability &
141  { return add(capability, std::string(value), doc); };
142 
143  // helper lambdas for adding capabilities
144  const auto have = [&](const std::string & capability, const std::string & doc) -> Capability &
145  { return add_bool(capability, true, doc + " is available."); };
146  const auto missing = [&](const std::string & capability,
147  const std::string & doc,
148  const std::string & help = "") -> Capability &
149  { return add_bool(capability, false, doc + " is not available. " + help); };
150 
151  const auto have_version = [&](const std::string & capability,
152  const std::string & doc,
153  const std::string & version) -> Capability &
154  { return add_string(capability, version, doc + " version " + version + " is available."); };
155  const auto petsc_missing = [&](const std::string & capability,
156  const std::string & doc) -> Capability &
157  {
158  return add_bool(
159  capability, false, doc + " is not available. Check your PETSc configure options.");
160  };
161  const auto libmesh_missing = [&](const std::string & capability,
162  const std::string & doc,
163  const std::string & config_option) -> Capability &
164  {
165  return add_bool(capability,
166  false,
167  doc + " is not available. It is controlled by the `" + config_option +
168  "` libMesh configure option.");
169  };
170 
171  {
172  const auto doc = "LibTorch machine learning and parallel tensor algebra library";
173 #ifdef MOOSE_LIBTORCH_ENABLED
174  add_string("libtorch", TORCH_VERSION, doc);
175 #else
176  missing("libtorch",
177  doc,
178  "Check "
179  "https://mooseframework.inl.gov/moose/getting_started/installation/"
180  "install_libtorch.html for "
181  "instructions on how to configure and build moose with libTorch.");
182 #endif
183  }
184 
185  {
186  const auto doc = "MFEM finite element library";
187 #ifdef MOOSE_MFEM_ENABLED
188  add_string("mfem", MFEM_VERSION_STRING, doc);
189 #else
190  missing("mfem",
191  doc,
192  "Install mfem using the scripts/update_and_rebuild_mfem.sh script after "
193  "first running scripts/update_and_rebuild_conduit.sh. Finally, configure "
194  "moose with ./configure --with-mfem");
195 #endif
196  }
197 
198  {
199  const auto doc = "New Engineering Material model Library, version 2";
200 #ifdef NEML2_ENABLED
201  have("neml2", doc);
202 #else
203  missing("neml2",
204  doc,
205  "Install neml2 using the scripts/update_and_rebuild_neml2.sh script, then "
206  "configure moose with ./configure --with-neml2 --with-libtorch");
207 #endif
208  }
209 
210  {
211  const auto doc = "gperftools code performance analysis and profiling library";
212 #ifdef HAVE_GPERFTOOLS
213  have("gperftools", doc);
214 #else
215  missing("gperftools",
216  doc,
217  "Check https://mooseframework.inl.gov/application_development/profiling.html "
218  "for instructions on profiling MOOSE based applications.");
219 #endif
220  }
221 
222  {
223  const auto doc = "libPNG portable network graphics format library";
224 #ifdef MOOSE_HAVE_LIBPNG
225  have("libpng", doc);
226 #else
227  missing("libpng",
228  doc,
229  "Install libpng through conda or your distribution and check that it gets "
230  "detected through pkg-config, then reconfigure and rebuild MOOSE.");
231 #endif
232  }
233 
234  {
235  const auto doc = "NVIDIA GPU parallel computing platform";
236 #ifdef PETSC_HAVE_CUDA
237  const std::string version = QUOTE(PETSC_PKG_CUDA_VERSION_MAJOR) "." QUOTE(
238  PETSC_PKG_CUDA_VERSION_MINOR) "." QUOTE(PETSC_PKG_CUDA_VERSION_SUBMINOR);
239  add_string("cuda", version, doc);
240 #else
241  missing("cuda", doc, "Add the CUDA bin directory to your path and rebuild PETSc.");
242 #endif
243  }
244 
245  {
246  const auto doc = "Kokkos performance portability programming ecosystem";
247 #ifdef MOOSE_KOKKOS_ENABLED
248  const std::string version = QUOTE(PETSC_PKG_KOKKOS_VERSION_MAJOR) "." QUOTE(
249  PETSC_PKG_KOKKOS_VERSION_MINOR) "." QUOTE(PETSC_PKG_KOKKOS_VERSION_SUBMINOR);
250  add_string("kokkos", version, doc);
251 #else
252  missing("kokkos",
253  doc,
254  "Rebuild PETSc with Kokkos support and libMesh. Then, reconfigure MOOSE with "
255  "--with-kokkos.");
256 #endif
257  }
258 
259  {
260  const auto doc = "Kokkos support for PETSc";
261 #ifdef PETSC_HAVE_KOKKOS
262  const std::string version = QUOTE(PETSC_PKG_KOKKOS_VERSION_MAJOR) "." QUOTE(
263  PETSC_PKG_KOKKOS_VERSION_MINOR) "." QUOTE(PETSC_PKG_KOKKOS_VERSION_SUBMINOR);
264  add_string("petsc_kokkos", version, doc);
265 #else
266  missing(
267  "petsc_kokkos", doc, "Rebuild PETSc with Kokkos support, then rebuild libMesh and MOOSE.");
268 #endif
269  }
270 
271  {
272  const auto doc = "Intel OneAPI XPU accelerator support";
273 #ifdef MOOSE_HAVE_XPU
274  if (torch::xpu::is_available())
275  have("xpu", doc);
276  else
277  missing("xpu", doc, "No usable XPU devices have been found.");
278 #else
279  missing("xpu", doc, "The torch version used to build this app has no XPU support.");
280 #endif
281  }
282 
283  add_int(
284  "ad_size",
285  MOOSE_AD_MAX_DOFS_PER_ELEM,
286  "MOOSE was configured and built with a dual number backing store size of " +
287  Moose::stringify(MOOSE_AD_MAX_DOFS_PER_ELEM) +
288  ". Complex simulations with many variables or contact problems may require larger "
289  "values. Reconfigure MOOSE with the --with-derivative-size=<n> option in the root of the "
290  "repository.")
291  .setExplicit();
292 
293  {
294  const std::string method = QUOTE(METHOD);
295  add_string("method", method, "The executable was built with METHOD=\"" + method + "\"")
296  .setExplicit()
297  .setEnumeration({"dbg", "devel", "oprof", "opt"});
298  }
299 
300  {
301  const std::string version = QUOTE(LIBMESH_DETECTED_PETSC_VERSION_MAJOR) "." QUOTE(
302  LIBMESH_DETECTED_PETSC_VERSION_MINOR) "." QUOTE(LIBMESH_DETECTED_PETSC_VERSION_SUBMINOR);
303  add_string("petsc", version, "Using PETSc version " + version + ".");
304  }
305 
306 #ifdef LIBMESH_PETSC_USE_DEBUG
307  add_bool("petsc_debug", true, "PETSc was built with debugging options.");
308 #else
309  add_bool("petsc_debug", false, "PETSc was built without debugging options.");
310 #endif
311 
312  {
313  const auto doc = "SuperLU direct solver";
314 #ifdef LIBMESH_PETSC_HAVE_SUPERLU_DIST
315  const std::string version = QUOTE(PETSC_PKG_SUPERLU_DIST_VERSION_MAJOR) "." QUOTE(
316  PETSC_PKG_SUPERLU_DIST_VERSION_MINOR) "." QUOTE(PETSC_PKG_SUPERLU_DIST_VERSION_SUBMINOR);
317  add_string("superlu", version, doc);
318 #else
319  petsc_missing("superlu", doc);
320 #endif
321  }
322 
323  {
324  const auto doc = "MUltifrontal Massively Parallel sparse direct Solver (MUMPS)";
325 #ifdef LIBMESH_PETSC_HAVE_MUMPS
326  const std::string version = QUOTE(PETSC_PKG_MUMPS_VERSION_MAJOR) "." QUOTE(
327  PETSC_PKG_MUMPS_VERSION_MINOR) "." QUOTE(PETSC_PKG_MUMPS_VERSION_SUBMINOR);
328  add_string("mumps", version, doc);
329 #else
330  petsc_missing("mumps", doc);
331 #endif
332  }
333 
334  {
335  const auto doc = "STRUMPACK - STRUctured Matrix PACKage solver library";
336 #ifdef LIBMESH_PETSC_HAVE_STRUMPACK
337  const std::string version = QUOTE(PETSC_PKG_STRUMPACK_VERSION_MAJOR) "." QUOTE(
338  PETSC_PKG_STRUMPACK_VERSION_MINOR) "." QUOTE(PETSC_PKG_STRUMPACK_VERSION_SUBMINOR);
339  add_string("strumpack", version, doc);
340 #else
341  petsc_missing("strumpack", doc);
342 #endif
343  }
344 
345  {
346  const auto doc = "Parmetis partitioning library";
347 #if defined(LIBMESH_PETSC_HAVE_PARMETIS)
348  const std::string version = QUOTE(PETSC_PKG_PARMETIS_VERSION_MAJOR) "." QUOTE(
349  PETSC_PKG_PARMETIS_VERSION_MINOR) "." QUOTE(PETSC_PKG_PARMETIS_VERSION_SUBMINOR);
350  add_string("parmetis", version, doc);
351 #elif defined(LIBMESH_HAVE_PARMETIS)
352  have("parmetis", doc);
353 #else
354  petsc_missing("parmetis", doc);
355 #endif
356  }
357 
358  {
359  const auto doc = "Chaco graph partitioning library";
360 #ifdef LIBMESH_PETSC_HAVE_CHACO
361  have("chaco", doc);
362 #else
363  petsc_missing("chaco", doc);
364 #endif
365  }
366 
367  {
368  const auto doc = "Party matrix or graph partitioning library";
369 #ifdef LIBMESH_PETSC_HAVE_PARTY
370  have("party", doc);
371 #else
372  petsc_missing("party", doc);
373 #endif
374  }
375 
376  {
377  const auto doc = "PT-Scotch graph partitioning library";
378 #ifdef LIBMESH_PETSC_HAVE_PTSCOTCH
379  const std::string version = QUOTE(PETSC_PKG_PTSCOTCH_VERSION_MAJOR) "." QUOTE(
380  PETSC_PKG_PTSCOTCH_VERSION_MINOR) "." QUOTE(PETSC_PKG_PTSCOTCH_VERSION_SUBMINOR);
381  add_string("ptscotch", version, doc);
382 #else
383  petsc_missing("ptscotch", doc);
384 #endif
385  }
386 
387  {
388  const auto doc = "Optimized BLAS library";
389 #ifdef PETSC_HAVE_OPENBLAS
390  const std::string version = QUOTE(PETSC_PKG_OPENBLAS_VERSION_MAJOR) "." QUOTE(
391  PETSC_PKG_OPENBLAS_VERSION_MINOR) "." QUOTE(PETSC_PKG_OPENBLAS_VERSION_SUBMINOR);
392  add_string("openblas", version, doc);
393 #else
394  petsc_missing("openblas", doc);
395 #endif
396  }
397 
398  {
399  const auto doc = "Umpire resource management library";
400 #ifdef PETSC_HAVE_UMPIRE
401  have("umpire", doc);
402 #else
403  petsc_missing("umpire", doc);
404 #endif
405  }
406 
407  {
408  const auto doc = "Scalable Library for Eigenvalue Problem Computations (SLEPc)";
409 #ifdef LIBMESH_HAVE_SLEPC
410  const auto version = QUOTE(LIBMESH_DETECTED_SLEPC_VERSION_MAJOR) "." QUOTE(
411  LIBMESH_DETECTED_SLEPC_VERSION_MINOR) "." QUOTE(LIBMESH_DETECTED_SLEPC_VERSION_SUBMINOR);
412  have_version("slepc", doc, version);
413 #else
414  petsc_missing("slepc", doc);
415 #endif
416  }
417 
418  {
419  const auto doc = "Exodus mesh file format library";
420 #ifdef LIBMESH_HAVE_EXODUS_API
421  const std::string version = QUOTE(LIBMESH_DETECTED_EXODUS_VERSION_MAJOR) "." QUOTE(
422  LIBMESH_DETECTED_EXODUS_VERSION_MINOR);
423  have_version("exodus", doc, version);
424 #else
425  libmesh_missing("exodus", doc, "--enable-exodus");
426 #endif
427  }
428 
429  {
430  const auto doc = "Netgen meshing library";
431 #ifdef LIBMESH_HAVE_NETGEN
432  const std::string version =
433  QUOTE(NETGEN_VERSION_MAJOR) "." QUOTE(NETGEN_VERSION_MINOR) "." QUOTE(NETGEN_VERSION_PATCH);
434  add_string("netgen", version, doc);
435 #else
436  libmesh_missing("netgen", doc, "--enable-netgen");
437 #endif
438  }
439 
440  {
441  const auto doc = "Visualization Toolkit (VTK)";
442 #ifdef LIBMESH_HAVE_VTK
443  const std::string version = QUOTE(LIBMESH_DETECTED_VTK_VERSION_MAJOR) "." QUOTE(
444  LIBMESH_DETECTED_VTK_VERSION_MINOR) "." QUOTE(LIBMESH_DETECTED_VTK_VERSION_SUBMINOR);
445  have_version("vtk", doc, version);
446 #else
447  libmesh_missing("vtk", doc, "--disable-vtk and --enable-vtk-required");
448 #endif
449  }
450 
451  {
452  const auto doc = "libcurl - the multiprotocol file transfer library";
453 #ifdef LIBMESH_HAVE_CURL
454  have("curl", doc);
455 #else
456  libmesh_missing("curl", doc, "--enable-curl");
457 #endif
458  }
459 
460  {
461  const auto doc = "Tecplot post-processing tools API";
462 #ifdef LIBMESH_HAVE_TECPLOT_API
463  have("tecplot", doc);
464 #else
465  libmesh_missing("tecplot", doc, "--enable-tecplot");
466 #endif
467  }
468 
469  {
470  const auto doc = "NVIDIA Tools Extension Library API";
471 #ifdef LIBMESH_HAVE_NVTX_API
472  have("nvtx_api", doc);
473 #else
474  libmesh_missing("nvtx_api", doc, "--with-nvtx");
475 #endif
476  }
477 
478  {
479  const auto doc = "libMesh performance logging";
480 #ifdef LIBMESH_ENABLE_PERFORMANCE_LOGGING
481  have("libmesh_perflog", doc);
482 #else
483  libmesh_missing("libmesh_perflog", doc, "--enable-perflog");
484 #endif
485  }
486 
487  {
488  const auto doc = "Boost C++ library";
489 #ifdef LIBMESH_HAVE_EXTERNAL_BOOST
490  have("boost", doc);
491 #else
492  libmesh_missing("boost", doc, "--with-boost");
493 #endif
494  }
495 
496  // libmesh stuff
497  {
498  const auto doc = "Adaptive mesh refinement";
499 #ifdef LIBMESH_ENABLE_AMR
500  have("amr", doc);
501 #else
502  libmesh_missing("amr", doc, "--disable-amr");
503 #endif
504  }
505 
506  {
507  const auto doc = "nanoflann library for Nearest Neighbor (NN) search with KD-trees";
508 #ifdef LIBMESH_HAVE_NANOFLANN
509  have("nanoflann", doc);
510 #else
511  libmesh_missing("nanoflann", doc, "--disable-nanoflann");
512 #endif
513  }
514 
515  {
516  const auto doc = "sfcurves library for space filling curves (required by geometric "
517  "partitioners such as SFCurves, Hilbert and Morton - not LGPL compatible)";
518 #ifdef LIBMESH_HAVE_SFCURVES
519  have("sfcurves", doc);
520 #else
521  libmesh_missing("sfcurves", doc, "--disable-sfc");
522 #endif
523  }
524 
525  {
526 #ifdef LIBMESH_HAVE_FPARSER
527 #ifdef LIBMESH_HAVE_FPARSER_JIT
528  const auto value = "jit";
529  const auto doc = "FParser enabled with just in time compilation support.";
530 #else
531  const auto value = "byte_code";
532  const auto doc = "FParser enabled.";
533 #endif
534  add_string("fparser", value, doc);
535 #else
536  add_bool("fparser",
537  false,
538  "FParser is disabled, libMesh was likely configured with --disable-fparser.");
539 #endif
540  }
541 
542 #ifdef LIBMESH_HAVE_DLOPEN
543  add_bool("dlopen", true, "The dlopen() system call is available to dynamically load libraries.");
544 #else
545  add_bool("dlopen",
546  false,
547  "The dlopen() system call is not available. Dynamic library loading is "
548  "not supported on this system.");
549 #endif
550 
551  {
552  const auto doc = "LibMesh support for threaded execution";
553 #ifdef LIBMESH_USING_THREADS
554  have("threads", doc);
555 #else
556  libmesh_missing("threads", doc, "--with-thread-model=tbb,pthread,openmp,auto,none");
557 #endif
558  }
559 
560  {
561  const auto doc = "OpenMP multi-platform shared-memory parallel programming API";
562 #ifdef LIBMESH_HAVE_OPENMP
563  have("openmp", doc);
564 #else
565  libmesh_missing("openmp", doc, "--with-thread-model=tbb,pthread,openmp,auto,none");
566 #endif
567  }
568  {
569  const auto doc = "POSIX Threads API";
570 #ifdef LIBMESH_HAVE_PTHREAD
571  have("pthread", doc);
572 #else
573  libmesh_missing("pthread", doc, "--with-thread-model=tbb,pthread,openmp,auto,none");
574 #endif
575  }
576  {
577  const auto doc = "oneAPI Threading Building Blocks (TBB) API";
578 #ifdef LIBMESH_HAVE_TBB_API
579  have("tbb", doc);
580 #else
581  libmesh_missing("tbb", doc, "--with-thread-model=tbb,pthread,openmp,auto,none");
582 #endif
583  }
584 
585  {
586  const auto doc = "libMesh unique ID support";
587 #ifdef LIBMESH_ENABLE_UNIQUE_ID
588  have("unique_id", doc);
589 #else
590  libmesh_missing("unique_id", doc, "--enable-unique-id");
591 #endif
592  }
593 
594  {
595 #ifdef LIBMESH_ENABLE_PARMESH
596  const auto value = "distributed";
597 #else
598  const auto value = "replicated";
599 #endif
600  add_string("mesh_mode", value, "libMesh default mesh mode")
601  .setExplicit()
602  .setEnumeration({"distributed", "replicated"});
603  }
604 
605  add_int("dof_id_bytes",
606  static_cast<int>(sizeof(dof_id_type)),
607  "Degree of freedom (DOF) identifiers use " + Moose::stringify(sizeof(dof_id_type)) +
608  " bytes for storage. This is controlled by the "
609  "--with-dof-id-bytes=<1|2|4|8> libMesh configure option.")
610  .setExplicit();
611 
612  {
613 #ifdef LIBMESH_HAVE_STATIC_LIBS
614  const auto value = "static";
615 #else
616  const auto value = "dynamic";
617 #endif
618  add_string("library_mode",
619  value,
620  "The libMesh library mode. This is controlled by the --enable-static libMesh "
621  "configure option.")
622  .setExplicit()
623  .setEnumeration({"dynamic", "static"});
624  }
625 
626  // compiler
627  {
628  const auto doc = "Compiler used to build the MOOSE framework.";
629 #if defined(__INTEL_LLVM_COMPILER)
630  const auto value = "intel";
631 #elif defined(__clang__)
632  const auto value = "clang";
633 #elif defined(__GNUC__) || defined(__GNUG__)
634  const auto value = "gcc";
635 #elif defined(_MSC_VER)
636  const auto value = "msvc";
637 #else
638  mooseDoOnce(mooseWarning("Failed to determine compiler; setting capability compiler=unknown"));
639  const auto value = "unknown";
640 #endif
641  add_string("compiler", value, doc)
642  .setExplicit()
643  .setEnumeration({"clang", "gcc", "intel", "msvc", "unknown"});
644  }
645 
646  // OS related
647  {
648 #ifdef __APPLE__
649  const auto value = "darwin";
650 #elif __WIN32__
651  const auto value = "win32";
652 #elif __linux__
653  const auto value = "linux";
654 #elif __unix__
655  const auto value = "unix";
656 #else
657  mooseDoOnce(mooseWarning("Failed to determine platform; setting capability platform=unknown"));
658  const auto value = "unknown";
659 #endif
660  add_string("platform", value, "Operating system this executable is running on.")
661  .setExplicit()
662  .setEnumeration({"darwin", "linux", "unix", "unknown", "win32"});
663  }
664 
665  // Installation type (in tree or installed)
666  {
667  // Try to find the path to the running executable
668  std::optional<std::string> executable_path;
669  {
670  Moose::ScopedThrowOnError scoped_throw_on_error;
671  try
672  {
673  executable_path = Moose::getExec();
674  }
675  catch (const MooseException &)
676  {
677  }
678  }
679 
680  std::string value = "unknown";
681 
682  if (executable_path)
683  {
684  // Try to follow all symlinks to get the real path
685  std::error_code ec;
686  const auto resolved_path =
687  std::filesystem::weakly_canonical(std::filesystem::path(*executable_path), ec);
688  if (ec)
689  mooseDoOnce(mooseWarning("Failed to resolve executable path '",
690  *executable_path,
691  "':\n",
692  ec.message(),
693  "\n\nSetting capability installation_type=unknown"));
694  else
695  {
696  // If the binary is in a folder "bin", we'll consider it installed.
697  // This isn't the best check, but it works with how we currently
698  // install applications in app.mk
699  value = resolved_path.parent_path().filename() == "bin" ? "relocated" : "in_tree";
700  }
701  }
702  else
703  mooseDoOnce(mooseWarning(
704  "Failed to determine executable path; setting capability installation_type=unknown"));
705 
706  add_string("installation_type", value, "The installation type of the application.")
707  .setExplicit()
708  .setEnumeration({"in_tree", "relocated", "unknown"});
709 
710  mooseAssert(isInstallationType(value), "Value mismatch");
711  }
712 }
713 
714 } // namespace Moose
std::string name(const ElemQuality q)
void registerMooseCapabilities()
Register the MOOSE capabilities.
Definition: Capabilities.C:127
static Capabilities & getCapabilities(const GetCapabilitiesPassKey)
Get the singleton Capabilities.
Definition: Capabilities.C:52
void mooseWarning(Args &&... args)
Emit a warning message with the given stringified, concatenated args.
Definition: MooseError.h:345
std::string dump() const
create a JSON dump of the capabilities registry
Definition: Capabilities.C:63
std::string getExec()
Gets the full path to the running executable on Mac OS X and linux.
void augment(const nlohmann::json &input, const AugmentPassKey)
Augment the capabilities with the given input capabilities.
Definition: Capabilities.C:80
std::variant< bool, int, std::string > Value
A capability can have a bool, int, or string value.
Definition: Capability.h:33
Holds the public (to MooseApp) facing CapabilityRegistry for storing and checking capabilities...
Definition: Capabilities.h:50
RegistryType _registry
Registry storage.
Real value(unsigned n, unsigned alpha, unsigned beta, Real x)
Capability & add(const std::string_view name, const Moose::Capability::Value &value, const std::string_view doc)
Add a capability.
An entry for a single capability.
Definition: Capability.h:29
std::string stringify(const T &t)
conversion to string
Definition: Conversion.h:64
std::string help
Common execption to be thrown when interacting with capabilities.
Provides a way for users to bail out of the current solve.
Registry of capabilities that checks capability requirements.
Scoped helper for setting Moose::_throw_on_error during this scope.
Definition: Moose.h:295
bool isInstallationType(const std::string &installation_type) const
Helper for isRelocated() and isInTree()
Definition: Capabilities.C:119
uint8_t dof_id_type