Source code for armi

# Copyright 2019 TerraPower, LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
Welcome to the Advanced Reactor Modeling Interface (ARMI).

This module initializes the ARMI platform. The bootstrap process is broken into several phases:

* Import fundamental dependencies in Python library and some third-party libs
* Investigate environment: Check Python version, check code version, check MPI situation,
  check for TTY/GUI/interactivity,
* Set up temp dirs
* Set up printout table formats (in preparation of logging info)
* Initialize all possible nuclide objects in the nuclide directory
* Discover and register available built-in :py:mod:`plugins <armi.plugins>`
  (custom ones registered after input available)
* Discover and define all potential configuration settings from available plugins
* Read input files
* Update :py:mod:`nuclide directory <armi.nucDirectory>` with depletion info based on config
* Discover and define all state :py:mod:`Parameters <armi.reactor.parameters>` on data model
  (possibly dependent on config)
* Discover :py:mod:`Entry points <armi.cli>` from plugins
* Choose entry point based on user command

If using the ``run`` entry point, additional work is done:

* Build :py:mod:`reactor model <armi.reactor>` based on input
* Build :py:mod:`operator object <armi.operators>` with specific calculation loop
* Build ordered interface stack based on configuration settings
* Begin looping over interface stack, operating upon data model according to operator design
* Loop until complete
* Wrap up
* Quit
"""
# ruff: noqa: F401
import atexit
import datetime
import importlib
import os
import signal
import subprocess
import sys
import traceback
import warnings
from typing import List, Optional, Type

import __main__ as main

# The _bootstrap module performs operations that may need to occur before it is
# necessarily safe to import the rest of the ARMI system. Things like:
# - configure the MPI environment
# - detect the nature of interaction with the user (terminal UI, GUI, unsupervized, etc)
# - Initialize the nuclide database
import armi._bootstrap
from armi import apps, cli, context, pluginManager, plugins, runLog
from armi.context import (
    APP_DATA,
    CURRENT_MODE,
    DOC,
    MPI_COMM,
    MPI_DISTRIBUTABLE,
    MPI_NODENAME,
    MPI_NODENAMES,
    MPI_RANK,
    MPI_SIZE,
    RES,
    ROOT,
    START_TIME,
    USER,
    Mode,
)
from armi.meta import __version__
from armi.nucDirectory import nuclideBases
from armi.reactor import flags, parameters

# ARMI does not configure its own application by default. This is mostly to catch issues
# involving calling code that requires the framework to be configured before that has
# explicitly taken place. An application should call `configure()` with its App class in
# order for ARMI to work properly
_app: Optional[apps.App] = None

_ARMI_CONFIGURE_CONTEXT: Optional[str] = None

# Advanced flag used in documentation builds to avoid isConfigured guards.
_ignoreConfigures = False


[docs]def disableFutureConfigures(): """Exposed function to ensure armi.configure() isn't called more than once.""" global _ignoreConfigures _ignoreConfigures = True
[docs]def isStableReleaseVersion(version=None): """Determine if the version should be considered a stable release.""" version = version or __version__ return "-" not in version
[docs]def init(choice=None, fName=None, cs=None, skipInspection=False): """ Scan a directory for armi inputs and load one to interact with. .. impl:: Settings are used to define an ARMI run. :id: I_ARMI_SETTING1 :implements: R_ARMI_SETTING This method initializes an ARMI run, and if successful returns an Operator. That operator is designed to drive the reactor simulation through time steps to simulate its operation. This method takes in a settings file or object to initialize the operator. Whether a settings file or object is supplied, the operator will be built based on the those settings. Because the total collection of settings can be modified by developers of ARMI applications, providing these settings allow ARMI end-users to define their simulation as granularly as they need. Parameters ---------- choice : int, optional Automatically run with this item out of the menu that would be produced by the existing YAML files. fName : str, optional The path to a settings file to load: my_case.yaml cs : Settings, optional If supplied, this CS object will supercede the other case input methods and use the object directly. skipInspection : bool, optional Whether or not the inputs should be checked for valid settings. Default is False. Examples -------- >>> o = armi.init() """ from armi import cases, settings if cs is None: if fName is None: fName = settings.promptForSettingsFile(choice) cs = settings.Settings(fName) armiCase = cases.Case(cs=cs) if not skipInspection: armiCase.checkInputs() try: return armiCase.initializeOperator() except: # Catch any and all errors. Naked exception on purpose. # Concatenate errors to the primary log file. runLog.close() raise
[docs]def getDefaultPlugins() -> List[Type[plugins.ArmiPlugin]]: """ Return a list containing the default set of ARMI Framework plugins. This is useful for an application to fold all of the ARMI Framework's capabilities into its own set of plugins. """ from armi import bookkeeping, cli, reactor from armi.physics import fuelCycle, neutronics, safety defaultPlugins = [ cli.EntryPointsPlugin, bookkeeping.BookkeepingPlugin, fuelCycle.FuelHandlerPlugin, neutronics.NeutronicsPlugin, safety.SafetyPlugin, reactor.ReactorPlugin, ] return defaultPlugins
[docs]def getDefaultPluginManager() -> pluginManager.ArmiPluginManager: """ Return a plugin manager containing the default set of ARMI Framework plugins. This is useful when using standalone facilities of ARMI without a specific application. """ pm = plugins.getNewPluginManager() for plugin in getDefaultPlugins(): pm.register(plugin) return pm
[docs]def isConfigured(): """Returns whether ARMI has been configured with an App.""" return _app is not None
[docs]def getPluginManager() -> Optional[pluginManager.ArmiPluginManager]: """Return the plugin manager, if there is one.""" global _app if _app is None: return None return _app.pluginManager
[docs]def getPluginManagerOrFail() -> pluginManager.ArmiPluginManager: """Return the plugin manager. Raise an error if there is none.""" global _app assert _app is not None, ( "The ARMI plugin manager was requested, no App has been configured. Ensure " "that `armi.configure()` has been called before attempting to interact with " "the plugin manager." ) return _app.pluginManager
[docs]def getApp() -> Optional[apps.App]: global _app return _app
def _cleanupOnCancel(signum, _frame): """Helper function to clean up upon cancellation.""" print( "Caught Cancel signal ({}); cleaning temporary files and exiting..." "".format(signum), file=sys.stderr, ) context.cleanTempDirs() sys.stdout.flush() sys.stderr.flush() sys.exit(1) # since we're handling the signal we have to cancel def _liveInterpreter(): """Return whether we are running within a live/interactive python interpreter.""" return not hasattr(main, "__file__")
[docs]def configure(app: Optional[apps.App] = None, permissive=False): """ Set the plugin manager for the Framework and configure internals to those plugins. Parameters ---------- app : An :py:class:`armi.apps.App` instance with which the framework is to be configured. If it is not provided, then the default ARMI App will be used. permissive : Whether or not an error should be produced if ``configure`` is called more than once. This should only be set to ``True`` under testing or demonstration purposes, where the contents of otherwise independent scripts need to be run under the same python instance. Important --------- Since this affects the behavior of several modules at their import time, it is generally not safe to re-configure the ARMI framework once it has been configured. Therefore this will raise an ``RuntimeError`` if such a re-configuration is attempted, unless ``permissive`` is set to ``True``. Notes ----- We are planning on encapsulating much of the global ARMI state that gets configured with an App into the App object itself (with some other things going into the Case object). This will provide a number of benefits, the main one being that it will become trivial to re-configure the framework, which is currently not possible. """ global _app global _ARMI_CONFIGURE_CONTEXT if _ignoreConfigures: return app = app or apps.App() if _app is not None: if permissive and isinstance(app, apps.App): return else: raise RuntimeError( "Multiple calls to armi.configure() are not allowed. " "Previous call from:\n{}".format(_ARMI_CONFIGURE_CONTEXT) ) assert not context.BLUEPRINTS_IMPORTED, ( "ARMI can no longer be configured after blueprints have been imported. " "Blueprints were imported from:\n{}".format(context.BLUEPRINTS_IMPORT_CONTEXT) ) _ARMI_CONFIGURE_CONTEXT = "".join(traceback.format_stack()) _app = app context.APP_NAME = app.name if _liveInterpreter(): runLog.LOG.startLog(name=f"interactive-{app.name}") cli.splash() pm = app.pluginManager parameters.collectPluginParameters(pm) parameters.applyAllParameters() _app.registerPluginFlags()
[docs]def applyAsyncioWindowsWorkaround() -> None: """ Apply Asyncio workaround for Windows and Python 3.8. This prevents a NotImplementedError on Windows with Python 3.8 his error showed up during jupyter notebook built-tests and documentation. See https://bugs.python.org/issue37373 """ import asyncio if ( sys.version_info[0] == 3 and sys.version_info[1] >= 8 and sys.platform.startswith("win") ): asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy())
applyAsyncioWindowsWorkaround() # The ``atexit`` handler is like putting it in a finally after everything. atexit.register(context.cleanTempDirs) # register cleanups upon HPC cancellations. Linux clusters will send a different signal. # SIGBREAK doesn't exist on non-windows # This actually doesn't work in mpi runs because MSMPI's mpiexec does not pass signals. if os.name == "nt": signal.signal(signal.SIGBREAK, _cleanupOnCancel) signal.signal(signal.SIGINT, _cleanupOnCancel)