Source code for armi.physics.neutronics.tests.test_crossSectionSettings

# 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.
"""XS Settings tests."""
import io
import unittest

from ruamel.yaml import YAML
import voluptuous as vol

from armi import settings
from armi.physics.neutronics.const import CONF_CROSS_SECTION
from armi.physics.neutronics.crossSectionSettings import CONF_BLOCK_REPRESENTATION
from armi.physics.neutronics.crossSectionSettings import CONF_GEOM
from armi.physics.neutronics.crossSectionSettings import XSModelingOptions
from armi.physics.neutronics.crossSectionSettings import XSSettingDef
from armi.physics.neutronics.crossSectionSettings import XSSettings
from armi.physics.neutronics.crossSectionSettings import xsSettingsValidator
from armi.physics.neutronics.tests.test_neutronicsPlugin import XS_EXAMPLE
from armi.physics.neutronics.settings import (
    CONF_XS_BLOCK_REPRESENTATION,
    CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION,
)
from armi.settings import caseSettings


[docs]class TestCrossSectionSettings(unittest.TestCase):
[docs] def test_crossSections(self): xsModel = XSModelingOptions( xsID="AA", geometry="0D", criticalBuckling=True, blockRepresentation="Median", ) self.assertEqual("AA", xsModel.xsID) self.assertEqual("0D", xsModel.geometry) self.assertEqual("Median", xsModel.blockRepresentation) self.assertFalse(xsModel.fluxIsPregenerated) self.assertFalse(xsModel.xsIsPregenerated) self.assertTrue(xsModel.criticalBuckling)
[docs] def test_pregeneratedCrossSections(self): cs = settings.Settings() xs = XSSettings() xa = XSModelingOptions("XA", xsFileLocation=["ISOXA"]) xs["XA"] = xa self.assertEqual(["ISOXA"], xa.xsFileLocation) self.assertNotIn("XB", xs) xs.setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) # Check that the file location of 'XB' still points to the same file location as 'XA'. self.assertEqual(xa, xs["XB"]) self.assertFalse(xa.fluxIsPregenerated) self.assertTrue(xa.xsIsPregenerated) self.assertFalse(xa.criticalBuckling)
[docs] def test_pregeneratedFluxInputs(self): xsModel = XSModelingOptions( xsID="AA", fluxFileLocation="ISOAA", geometry="0D", criticalBuckling=True, blockRepresentation="Median", ) self.assertEqual("AA", xsModel.xsID) self.assertEqual("0D", xsModel.geometry) self.assertEqual("ISOAA", xsModel.fluxFileLocation) self.assertTrue(xsModel.fluxIsPregenerated) self.assertTrue(xsModel.criticalBuckling) self.assertEqual("Median", xsModel.blockRepresentation)
[docs] def test_prioritization(self): xsModel = XSModelingOptions( xsID="AA", geometry="0D", criticalBuckling=True, xsPriority=2, xsExecuteExclusive=True, ) self.assertEqual("AA", xsModel.xsID) self.assertEqual(True, xsModel.xsExecuteExclusive) self.assertEqual(2, xsModel.xsPriority) xsModel = XSModelingOptions( xsID="AA", geometry="0D", criticalBuckling=True, ) # defaults work xsModel.setDefaults("Average", False) self.assertEqual(False, xsModel.xsExecuteExclusive) self.assertEqual(5, xsModel.xsPriority)
[docs] def test_homogeneousXsDefaultSettingAssignment(self): """ Make sure the object can whip up an unspecified xsID by default. This is used when user hasn't specified anything. """ cs = settings.Settings() xsModel = XSSettings() xsModel.setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertNotIn("YA", xsModel) self.assertEqual(xsModel["YA"].geometry, "0D") self.assertEqual(xsModel["YA"].criticalBuckling, True) self.assertEqual(xsModel["YA"].ductHeterogeneous, False) self.assertEqual(xsModel["YA"].traceIsotopeThreshold, 0.0)
[docs] def test_setDefaultSettingsByLowestEnvGroupHomogeneous(self): # Initialize some micro suffix in the cross sections cs = settings.Settings() xs = XSSettings() jd = XSModelingOptions("JD", geometry="0D", criticalBuckling=False) xs["JD"] = jd xs.setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertIn("JD", xs) # Check that new micro suffix `JF` with higher burn-up group gets assigned the same settings as `JD` self.assertNotIn("JF", xs) self.assertEqual(xs["JD"], xs["JF"]) self.assertNotIn("JF", xs) # Check that new micro suffix `JG` with higher burn-up group gets assigned the same settings as `JD` self.assertNotIn("JG", xs) self.assertEqual(xs["JG"], xs["JD"]) # Check that new micro suffix `JB` with lower burn-up group does NOT get assigned the same settings as `JD` self.assertNotIn("JB", xs) self.assertNotEqual(xs["JD"], xs["JB"])
[docs] def test_setDefaultSettingsByLowestEnvGroupOneDimensional(self): # Initialize some micro suffix in the cross sections cs = settings.Settings() xsModel = XSSettings() rq = XSModelingOptions( "RQ", geometry="1D cylinder", blockRepresentation="ComponentAverage1DCylinder", meshSubdivisionsPerCm=1.0, ) xsModel["RQ"] = rq xsModel.setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) # Check that new micro suffix `RY` with higher burn-up group gets assigned the same settings as `RQ` self.assertNotIn("RY", xsModel) self.assertEqual(xsModel["RY"], xsModel["RQ"]) # Check that new micro suffix `RZ` with higher burn-up group gets assigned the same settings as `RQ` self.assertNotIn("RZ", xsModel) self.assertEqual(xsModel["RZ"], xsModel["RQ"]) # Check that new micro suffix `RA` with lower burn-up group does NOT get assigned the same settings as `RQ` self.assertNotIn("RA", xsModel) self.assertNotEqual(xsModel["RA"], xsModel["RQ"])
[docs] def test_optionalKey(self): """Test that optional key shows up with default value.""" cs = settings.Settings() xsModel = XSSettings() da = XSModelingOptions( "DA", geometry="1D cylinder", meshSubdivisionsPerCm=1.0, ductHeterogeneous=True, traceIsotopeThreshold=1.0e-5, ) xsModel["DA"] = da xsModel.setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertEqual(xsModel["DA"].mergeIntoClad, ["gap"]) self.assertEqual(xsModel["DA"].meshSubdivisionsPerCm, 1.0) self.assertEqual(xsModel["DA"].ductHeterogeneous, True) self.assertEqual(xsModel["DA"].traceIsotopeThreshold, 1.0e-5) self.assertEqual(xsModel["DA"].mergeIntoFuel, [])
[docs] def test_badCrossSections(self): with self.assertRaises(TypeError): # This will fail because it is not the required # Dict[str: Dict] structure xsSettingsValidator({CONF_GEOM: "4D"}) with self.assertRaises(vol.error.MultipleInvalid): # This will fail because it has an invalid type for ``driverID`` xsSettingsValidator({"AA": {"driverId": 0.0}}) with self.assertRaises(vol.error.MultipleInvalid): # This will fail because it has an invalid value for # the ``blockRepresentation`` xsSettingsValidator({"AA": {CONF_BLOCK_REPRESENTATION: "Invalid"}}) with self.assertRaises(vol.error.MultipleInvalid): # This will fail because the ``xsID`` is not one or two # characters xsSettingsValidator({"AAA": {CONF_BLOCK_REPRESENTATION: "Average"}})
[docs]class TestXSSettings(unittest.TestCase):
[docs] def test_yamlIO(self): """Ensure we can read/write this custom setting object to yaml.""" yaml = YAML() inp = yaml.load(io.StringIO(XS_EXAMPLE)) xs = XSSettingDef("TestSetting") xs.setValue(inp) self.assertEqual(xs.value["BA"].geometry, "1D slab") outBuf = io.StringIO() output = xs.dump() yaml.dump(output, outBuf) outBuf.seek(0) inp2 = yaml.load(outBuf) self.assertEqual(inp.keys(), inp2.keys())
[docs] def test_caseSettings(self): """ Test the setting of the cross section setting using the case settings object. Notes ----- The purpose of this test is to ensure that the cross sections sections can be removed from an existing case settings object once they have been set. """ def _setInitialXSSettings(): cs = caseSettings.Settings() cs[CONF_CROSS_SECTION] = XSSettings() cs[CONF_CROSS_SECTION]["AA"] = XSModelingOptions("AA", geometry="0D") cs[CONF_CROSS_SECTION]["BA"] = XSModelingOptions("BA", geometry="0D") self.assertIn("AA", cs[CONF_CROSS_SECTION]) self.assertIn("BA", cs[CONF_CROSS_SECTION]) self.assertNotIn("CA", cs[CONF_CROSS_SECTION]) self.assertNotIn("DA", cs[CONF_CROSS_SECTION]) return cs cs = _setInitialXSSettings() cs[CONF_CROSS_SECTION] = {"AA": {}, "BA": {}} self.assertDictEqual(cs[CONF_CROSS_SECTION], {}) self.assertTrue(isinstance(cs[CONF_CROSS_SECTION], XSSettings)) # Produce an error if the setting is set to # a None value cs = _setInitialXSSettings() with self.assertRaises(TypeError): cs[CONF_CROSS_SECTION] = None cs = _setInitialXSSettings() cs[CONF_CROSS_SECTION] = {"AA": None, "BA": {}} self.assertDictEqual(cs[CONF_CROSS_SECTION], {}) # Test that a new XS setting can be added to an existing # caseSetting using the ``XSModelingOptions`` or using # a dictionary. cs = _setInitialXSSettings() cs[CONF_CROSS_SECTION].update( {"CA": XSModelingOptions("CA", geometry="0D"), "DA": {CONF_GEOM: "0D"}} ) self.assertIn("AA", cs[CONF_CROSS_SECTION]) self.assertIn("BA", cs[CONF_CROSS_SECTION]) self.assertIn("CA", cs[CONF_CROSS_SECTION]) self.assertIn("DA", cs[CONF_CROSS_SECTION]) # Clear out the settings by setting the value to a None. # This will be interpreted as a empty dictionary. cs[CONF_CROSS_SECTION] = {} self.assertDictEqual(cs[CONF_CROSS_SECTION], {}) self.assertTrue(isinstance(cs[CONF_CROSS_SECTION], XSSettings)) # This will fail because the ``setDefaults`` method on the # ``XSSettings`` has not yet been called. with self.assertRaises(ValueError): cs[CONF_CROSS_SECTION]["AA"] cs[CONF_CROSS_SECTION].setDefaults( blockRepresentation=cs[CONF_XS_BLOCK_REPRESENTATION], validBlockTypes=cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) cs[CONF_CROSS_SECTION]["AA"] self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].geometry, "0D")
[docs] def test_csBlockRepresentation(self): """ Test that the XS block representation is applied globally, but only to XS modeling options where the blockRepresentation has not already been assigned. """ cs = caseSettings.Settings() cs[CONF_XS_BLOCK_REPRESENTATION] = "FluxWeightedAverage" cs[CONF_CROSS_SECTION] = XSSettings() cs[CONF_CROSS_SECTION]["AA"] = XSModelingOptions("AA", geometry="0D") cs[CONF_CROSS_SECTION]["BA"] = XSModelingOptions( "BA", geometry="0D", blockRepresentation="Average" ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].blockRepresentation, None) self.assertEqual(cs[CONF_CROSS_SECTION]["BA"].blockRepresentation, "Average") cs[CONF_CROSS_SECTION].setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertEqual( cs[CONF_CROSS_SECTION]["AA"].blockRepresentation, "FluxWeightedAverage" ) self.assertEqual(cs[CONF_CROSS_SECTION]["BA"].blockRepresentation, "Average")
[docs] def test_csBlockRepresentationFileLocation(self): """ Test that default blockRepresentation is applied correctly to a XSModelingOption that has the ``xsFileLocation`` attribute defined. """ cs = caseSettings.Settings() cs[CONF_XS_BLOCK_REPRESENTATION] = "FluxWeightedAverage" cs[CONF_CROSS_SECTION] = XSSettings() cs[CONF_CROSS_SECTION]["AA"] = XSModelingOptions("AA", xsFileLocation=[]) # Check FluxWeightedAverage cs[CONF_CROSS_SECTION].setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertEqual( cs[CONF_CROSS_SECTION]["AA"].blockRepresentation, "FluxWeightedAverage" ) # Check Average cs[CONF_XS_BLOCK_REPRESENTATION] = "Average" cs[CONF_CROSS_SECTION]["AA"] = XSModelingOptions("AA", xsFileLocation=[]) cs[CONF_CROSS_SECTION].setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].blockRepresentation, "Average") # Check Median cs[CONF_XS_BLOCK_REPRESENTATION] = "Average" cs[CONF_CROSS_SECTION]["AA"] = XSModelingOptions( "AA", xsFileLocation=[], blockRepresentation="Median" ) cs[CONF_CROSS_SECTION].setDefaults( cs[CONF_XS_BLOCK_REPRESENTATION], cs[CONF_DISABLE_BLOCK_TYPE_EXCLUSION_IN_XS_GENERATION], ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].blockRepresentation, "Median")
[docs] def test_xsSettingsSetDefault(self): """Test the configuration options of the ``setDefaults`` method.""" cs = caseSettings.Settings() cs[CONF_XS_BLOCK_REPRESENTATION] = "FluxWeightedAverage" cs[CONF_CROSS_SECTION].setDefaults( blockRepresentation=cs[CONF_XS_BLOCK_REPRESENTATION], validBlockTypes=None ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].validBlockTypes, None) cs[CONF_CROSS_SECTION].setDefaults( blockRepresentation=cs[CONF_XS_BLOCK_REPRESENTATION], validBlockTypes=True ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].validBlockTypes, None) cs[CONF_CROSS_SECTION].setDefaults( blockRepresentation=cs[CONF_XS_BLOCK_REPRESENTATION], validBlockTypes=False ) self.assertEqual(cs[CONF_CROSS_SECTION]["AA"].validBlockTypes, ["fuel"]) cs[CONF_CROSS_SECTION].setDefaults( blockRepresentation=cs[CONF_XS_BLOCK_REPRESENTATION], validBlockTypes=["control", "fuel", "plenum"], ) self.assertEqual( cs[CONF_CROSS_SECTION]["AA"].validBlockTypes, ["control", "fuel", "plenum"] )