Source code for armi.physics.neutronics.latticePhysics.tests.test_latticeInterface

# Copyright 2021 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.

"""Test the Lattice Interface."""
from collections import OrderedDict
import unittest

from armi.physics.neutronics.latticePhysics.latticePhysicsInterface import (
    LatticePhysicsInterface,
)
from armi import settings
from armi.nuclearDataIO.cccc import isotxs
from armi.operators.operator import Operator
from armi.physics.neutronics import LatticePhysicsFrequency
from armi.physics.neutronics.crossSectionGroupManager import CrossSectionGroupManager
from armi.physics.neutronics.settings import CONF_GEN_XS
from armi.physics.neutronics.settings import CONF_GLOBAL_FLUX_ACTIVE
from armi.reactor.reactors import Reactor, Core
from armi.reactor.tests.test_blocks import buildSimpleFuelBlock
from armi.tests import mockRunLogs
from armi.reactor.assemblies import (
    HexAssembly,
    grids,
)
from armi.tests import ISOAA_PATH

# As an interface, LatticePhysicsInterface must be subclassed to be used
[docs]class LatticeInterfaceTester(LatticePhysicsInterface): def __init__(self, r, cs): self.name = "LatticeInterfaceTester" super().__init__(r, cs) def _getExecutablePath(self): return "/tmp/fake_path"
[docs] def readExistingXSLibraries(self, cycle, node): pass
[docs]class LatticeInterfaceTesterLibFalse(LatticeInterfaceTester): """Subclass setting _newLibraryShouldBeCreated = False.""" def _newLibraryShouldBeCreated(self, cycle, representativeBlockList, xsIDs): self.testVerification = True return False
[docs]class TestLatticePhysicsInterfaceBase(unittest.TestCase): @classmethod def setUpClass(cls): # create empty reactor core cls.o = Operator(settings.Settings()) cls.o.r = Reactor("testReactor", None) cls.o.r.core = Core("testCore") # add an assembly with a single block cls.assembly = HexAssembly("testAssembly") cls.assembly.spatialGrid = grids.AxialGrid.fromNCells(1) cls.assembly.spatialGrid.armiObject = cls.assembly cls.assembly.add(buildSimpleFuelBlock()) # cls.o.r.core.add(assembly) # init and add interfaces cls.xsGroupInterface = CrossSectionGroupManager(cls.o.r, cls.o.cs) cls.o.addInterface(cls.xsGroupInterface)
[docs]class TestLatticePhysicsInterface(TestLatticePhysicsInterfaceBase): """Test Lattice Physics Interface.""" @classmethod def setUpClass(cls): super().setUpClass() cls.latticeInterface = LatticeInterfaceTesterLibFalse(cls.o.r, cls.o.cs) cls.o.addInterface(cls.latticeInterface) def setUp(self): self.o.r.core.lib = "Nonsense" self.latticeInterface.testVerification = False
[docs] def test_includeGammaXS(self): """Test that we can correctly flip the switch to calculate gamma XS. .. test:: Users can flip a setting to determine if gamma XS are generated. :id: T_ARMI_GAMMA_XS :tests: R_ARMI_GAMMA_XS """ # The default operator here turns off Gamma XS generation self.assertFalse(self.latticeInterface.includeGammaXS) self.assertEqual(self.o.cs[CONF_GLOBAL_FLUX_ACTIVE], "Neutron") # but we can create an operator that turns on Gamma XS generation cs = settings.Settings().modified( newSettings={CONF_GLOBAL_FLUX_ACTIVE: "Neutron and Gamma"} ) newOperator = Operator(cs) newLatticeInterface = LatticeInterfaceTesterLibFalse(newOperator.r, cs) self.assertTrue(newLatticeInterface.includeGammaXS) self.assertEqual(cs[CONF_GLOBAL_FLUX_ACTIVE], "Neutron and Gamma")
[docs] def test_latticePhysicsInterface(self): """Super basic test of the LatticePhysicsInterface.""" self.assertEqual(self.latticeInterface._updateBlockNeutronVelocities, True) self.assertEqual(self.latticeInterface.executablePath, "/tmp/fake_path") self.assertEqual(self.latticeInterface.executableRoot, "/tmp") self.latticeInterface.updateXSLibrary(0) self.assertEqual(len(self.latticeInterface._oldXsIdsAndBurnup), 0)
[docs] def test_interactBOL(self): """ Test interactBOL() with different update frequencies. Notes ----- Unlike other interactions, self.o.r.core.lib is not set to None by the BOC interaction, so this test does not have a good means of verifying the correct function, so we use self.testVerification instead. """ self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.never self.latticeInterface.interactBOL() self.assertFalse(self.latticeInterface.testVerification) self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.everyNode ) self.latticeInterface.interactBOL() self.assertFalse(self.latticeInterface.testVerification) self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.BOL self.latticeInterface.interactBOL() self.assertTrue(self.latticeInterface.testVerification)
[docs] def test_interactBOC(self): """ Test interactBOC() with different update frequencies. Notes ----- Unlike other interactions, self.o.r.core.lib is not set to None by the BOC interaction, so this test does not have a good means of verifying the correct function, so we use self.testVerification instead. """ self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.BOL self.latticeInterface.interactBOC() self.assertFalse(self.latticeInterface.testVerification) self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.everyNode ) self.latticeInterface.interactBOC() self.assertFalse(self.latticeInterface.testVerification) self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.BOC self.latticeInterface.interactBOC() self.assertTrue(self.latticeInterface.testVerification)
[docs] def test_interactEveryNode(self): """Test interactEveryNode() with different update frequencies.""" self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.BOC self.latticeInterface.interactEveryNode() self.assertEqual(self.o.r.core.lib, "Nonsense") self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.everyNode ) self.latticeInterface.interactEveryNode() self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactEveryNodeWhenCoupled(self): """ Test that the XS lib is not cleared when coupled iterations are turned on and XS will be generated during the coupled iterations. """ self.o.couplingIsActive = lambda: True self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.firstCoupledIteration ) self.latticeInterface.interactEveryNode() self.assertEqual(self.o.r.core.lib, "Nonsense") self.o.couplingIsActive = lambda: False self.latticeInterface.interactEveryNode() self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactEveryNodeWhenCoupledButNot(self): """ Test that the XS lib is cleared when coupled iterations are turned on but the lattice physics frequency is not high enough. """ self.o.couplingIsActive = lambda: True self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.firstCoupledIteration ) self.latticeInterface.interactEveryNode() self.assertEqual(self.o.r.core.lib, "Nonsense") self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.everyNode ) self.latticeInterface.interactEveryNode() self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactEveryNodeFirstCoupled(self): """Test interactEveryNode() with LatticePhysicsFrequency.firstCoupledIteration.""" self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.firstCoupledIteration ) self.latticeInterface.interactEveryNode() self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactEveryNodeAll(self): """Test interactEveryNode() with LatticePhysicsFrequency.all.""" self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.all self.latticeInterface.interactEveryNode() self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactFirstCoupledIteration(self): """Test interactCoupled() with different update frequencies on first iteration.""" self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.everyNode ) self.latticeInterface.interactCoupled(iteration=0) self.assertEqual(self.o.r.core.lib, "Nonsense") self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.firstCoupledIteration ) self.latticeInterface.interactCoupled(iteration=0) self.assertIsNone(self.o.r.core.lib)
[docs] def test_interactAll(self): """Test interactCoupled() with different update frequencies on non-first iteration.""" self.latticeInterface._latticePhysicsFrequency = ( LatticePhysicsFrequency.firstCoupledIteration ) self.latticeInterface.interactCoupled(iteration=1) self.assertEqual(self.o.r.core.lib, "Nonsense") self.latticeInterface._latticePhysicsFrequency = LatticePhysicsFrequency.all self.latticeInterface.interactCoupled(iteration=1) self.assertIsNone(self.o.r.core.lib)
[docs] def test_getSuffix(self): self.assertEqual(self.latticeInterface._getSuffix(7), "")
[docs]class TestLatticePhysicsLibraryCreation(TestLatticePhysicsInterfaceBase): """Test variations of _newLibraryShouldBeCreated.""" @classmethod def setUpClass(cls): super().setUpClass() cls.latticeInterface = LatticeInterfaceTester(cls.o.r, cls.o.cs) cls.o.addInterface(cls.latticeInterface) cls.xsGroupInterface.representativeBlocks = OrderedDict({"AA": cls.assembly[0]}) cls.b, cls.xsIDs = cls.latticeInterface._getBlocksAndXsIds() def setUp(self): """Reset representativeBlocks and CONF_GEN_XS.""" self.xsGroupInterface.representativeBlocks = OrderedDict( {"AA": self.assembly[0]} ) self.assembly[0].p.xsType = "A" self.o.cs[CONF_GEN_XS] = "" self.o.r.core.lib = isotxs.readBinary(ISOAA_PATH)
[docs] def test_libCreation_NoGenXS(self): """No ISOTXS and xs gen not requested.""" self.o.r.core.lib = None with mockRunLogs.BufferLog() as mock: xsGen = self.latticeInterface._newLibraryShouldBeCreated( 1, self.b, self.xsIDs ) self.assertIn( "Cross sections will not be generated on cycle 1.", mock.getStdout() ) self.assertFalse(xsGen)
[docs] def test_libCreation_GenXS(self): """No ISOTXS and xs gen requested.""" self.o.cs[CONF_GEN_XS] = "Neutron" self.o.r.core.lib = None with mockRunLogs.BufferLog() as mock: xsGen = self.latticeInterface._newLibraryShouldBeCreated( 1, self.b, self.xsIDs ) self.assertIn( "Cross sections will be generated on cycle 1 for the following XS IDs: ['AA']", mock.getStdout(), ) self.assertTrue(xsGen)
[docs] def test_libCreation_NoGenXS_2(self): """ISOTXS present and has all of the necessary information.""" with mockRunLogs.BufferLog() as mock: xsGen = self.latticeInterface._newLibraryShouldBeCreated( 1, self.b, self.xsIDs ) self.assertIn( "The generation of XS will be skipped.", mock.getStdout(), ) self.assertFalse(xsGen)
[docs] def test_libCreation_GenXS_2(self): """ISOTXS present and does not have all of the necessary information.""" self.xsGroupInterface.representativeBlocks = OrderedDict( {"BB": self.assembly[0]} ) b, xsIDs = self._modifyXSType() with mockRunLogs.BufferLog() as mock: xsGen = self.latticeInterface._newLibraryShouldBeCreated(1, b, xsIDs) self.assertIn( "is not enabled, but will be run to generate these missing cross sections.", mock.getStdout(), ) self.assertTrue(xsGen)
[docs] def test_libCreation_GenXS_3(self): """ISOTXS present and does not have all of the necessary information.""" self.o.cs[CONF_GEN_XS] = "Neutron" b, xsIDs = self._modifyXSType() with mockRunLogs.BufferLog() as mock: xsGen = self.latticeInterface._newLibraryShouldBeCreated(1, b, xsIDs) self.assertIn("These will be generated on cycle ", mock.getStdout()) self.assertTrue(xsGen)
def _modifyXSType(self): self.xsGroupInterface.representativeBlocks = OrderedDict( {"BB": self.assembly[0]} ) self.assembly[0].p.xsType = "B" return self.latticeInterface._getBlocksAndXsIds()