Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement more detailed __lt__ for components. #1882

Draft
wants to merge 39 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
707aa88
Implement more detailed __lt__ for components.
mgjarrett Sep 11, 2024
35b63a8
Add a unit test for component sorting.
mgjarrett Sep 16, 2024
6bc7dc1
Update release notes.
mgjarrett Sep 16, 2024
1265d7c
Address linting error
mgjarrett Sep 16, 2024
dbed10f
Catch NotImplementedError
mgjarrett Sep 16, 2024
0fd1632
Remove all of the circular imports in components.
mgjarrett Sep 16, 2024
cca4a14
Fix broken unit tests.
mgjarrett Sep 16, 2024
06f1ec1
Fix unit test reference result.
mgjarrett Sep 16, 2024
37e8f0b
Register RadialSegment component type.
mgjarrett Sep 16, 2024
af44807
Black formatting.
mgjarrett Sep 16, 2024
c65a0c3
Merge branch 'main' into component_sort
mgjarrett Sep 16, 2024
9cb98e5
Refactor more unit tests.
mgjarrett Sep 16, 2024
d50b6dc
Ignore unused import
mgjarrett Sep 16, 2024
d1db895
Revert "Refactor more unit tests."
mgjarrett Sep 16, 2024
6c891fd
Revert "Fix broken unit tests."
mgjarrett Sep 16, 2024
914fe58
Revert "Remove all of the circular imports in components."
mgjarrett Sep 16, 2024
88feb80
Clear merge conflict.
mgjarrett Sep 16, 2024
0b1b6ef
Revert trivial change.
mgjarrett Sep 16, 2024
12980cd
Add getCircleInnerDiameter for UnshapedComponent.
mgjarrett Sep 16, 2024
a62fd41
Fix circular import.
mgjarrett Sep 16, 2024
acbf2fc
Black formatting.
mgjarrett Sep 16, 2024
53780e4
Update docstring.
mgjarrett Sep 16, 2024
2a7e4a1
Address review comments.
mgjarrett Sep 16, 2024
64d1d09
One more review comment.
mgjarrett Sep 16, 2024
37e4310
Remove commented code.
mgjarrett Sep 16, 2024
80d57ab
Remove kwargs
mgjarrett Sep 16, 2024
9457b49
Black formatting
mgjarrett Sep 16, 2024
6dca45a
Fix namespace errors.
mgjarrett Sep 16, 2024
4e2234b
Add getCircleInnerDiameter to RadialSegment.
mgjarrett Sep 16, 2024
48603a4
Update armi/reactor/components/__init__.py
mgjarrett Sep 16, 2024
3ccdc5b
Update doc/release/0.4.rst
mgjarrett Sep 16, 2024
2075f3a
Revert to main blocks.py
mgjarrett Sep 16, 2024
5b601f9
Revert import change.
mgjarrett Sep 16, 2024
15155be
Merge branch 'component_sort' of https://github.com/terrapower/armi i…
mgjarrett Sep 16, 2024
7733ad8
Revert more import changes.
mgjarrett Sep 16, 2024
9b61363
Revert more imports.
mgjarrett Sep 16, 2024
b87c1d1
Fix an import.
mgjarrett Sep 16, 2024
557a333
Merge branch 'main' into component_sort
john-science Oct 23, 2024
0fb4219
merging in main
john-science Nov 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 7 additions & 5 deletions armi/reactor/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@
See :doc:`/developer/index`.
"""

from typing import Dict, Callable, Union, TYPE_CHECKING
from typing import TYPE_CHECKING, Callable, Dict, Union
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved

from armi import plugins

Expand All @@ -66,9 +66,11 @@ class ReactorPlugin(plugins.ArmiPlugin):
@staticmethod
@plugins.HOOKIMPL
def defineBlockTypes():
from armi.reactor.components.basicShapes import Rectangle, Hexagon
from armi.reactor.components.volumetricShapes import RadialSegment
from armi.reactor import blocks
from armi.reactor.components.basicShapes import Hexagon, Rectangle
from armi.reactor.components.volumetricShapes import (
RadialSegment,
)

return [
(Rectangle, blocks.CartesianBlock),
Expand All @@ -79,8 +81,8 @@ def defineBlockTypes():
@staticmethod
@plugins.HOOKIMPL
def defineAssemblyTypes():
from armi.reactor.blocks import HexBlock, CartesianBlock, ThRZBlock
from armi.reactor.assemblies import HexAssembly, CartesianAssembly, ThRZAssembly
from armi.reactor.blocks import CartesianBlock, HexBlock, ThRZBlock
from armi.reactor.assemblies import CartesianAssembly, HexAssembly, ThRZAssembly

return [
(HexBlock, HexAssembly),
Expand Down
46 changes: 22 additions & 24 deletions armi/reactor/blocks.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,29 +18,31 @@

Assemblies are made of blocks. Blocks are made of components.
"""
from typing import Optional, Type, Tuple, ClassVar
import collections
import copy
import math
from typing import ClassVar, Optional, Tuple, Type

import numpy as np

from armi import nuclideBases
from armi import runLog
from armi import nuclideBases, runLog
from armi.bookkeeping import report
from armi.nucDirectory import elements
from armi.nuclearDataIO import xsCollections
from armi.physics.neutronics import GAMMA
from armi.physics.neutronics import NEUTRON
from armi.reactor import blockParameters
from armi.reactor import components
from armi.reactor import composites
from armi.reactor import geometry
from armi.reactor import grids
from armi.reactor import parameters
from armi.reactor.components import basicShapes
from armi.reactor.components.basicShapes import Hexagon, Circle
from armi.reactor.components.complexShapes import Helix
from armi.physics.neutronics import GAMMA, NEUTRON
from armi.reactor import (
blockParameters,
components,
composites,
geometry,
grids,
parameters,
)
from armi.reactor.components.basicShapes import Circle, Hexagon, Rectangle
from armi.reactor.components.complexShapes import Helix, HoledHexagon

# this unused import is required so that the RadialSegment component type is registered
from armi.reactor.components.volumetricShapes import RadialSegment
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved
from armi.reactor.flags import Flags
from armi.reactor.parameters import ParamLocation
from armi.utils import densityTools
Expand Down Expand Up @@ -232,7 +234,7 @@ def getSmearDensity(self, cold=True):
if not fuels:
return 0.0 # Smear density is not computed for non-fuel blocks

circles = self.getComponentsOfShape(components.Circle)
circles = self.getComponentsOfShape(Circle)
if not circles:
raise ValueError(
"Cannot get smear density of {}. There are no circular components.".format(
Expand Down Expand Up @@ -1066,11 +1068,7 @@ def getNumPins(self):
nPins = [
sum(
[
(
int(c.getDimension("mult"))
if isinstance(c, basicShapes.Circle)
else 0
)
(int(c.getDimension("mult")) if isinstance(c, Circle) else 0)
for c in self.iterComponents(compType)
]
)
Expand Down Expand Up @@ -1756,7 +1754,7 @@ class HexBlock(Block):
pitch, pin linear power densities, hydraulic diameter, and retrieving inner and outer pitch.
"""

PITCH_COMPONENT_TYPE: ClassVar[_PitchDefiningComponent] = (components.Hexagon,)
PITCH_COMPONENT_TYPE: ClassVar[_PitchDefiningComponent] = (Hexagon,)

def __init__(self, name, height=1.0):
Block.__init__(self, name, height)
Expand Down Expand Up @@ -2246,11 +2244,11 @@ def getPinToDuctGap(self, cold=False):
duct = None
if any(ducts):
duct = ducts[0]
if not isinstance(duct, components.Hexagon):
if not isinstance(duct, Hexagon):
# getPinCenterFlatToFlat only works for hexes
# inner most duct might be circle or some other shape
duct = None
elif isinstance(duct, components.HoledHexagon):
elif isinstance(duct, HoledHexagon):
# has no ip and is circular on inside so following
# code will not work
duct = None
Expand Down Expand Up @@ -2596,7 +2594,7 @@ def getHydraulicDiameter(self):

class CartesianBlock(Block):
PITCH_DIMENSION = "widthOuter"
PITCH_COMPONENT_TYPE = components.Rectangle
PITCH_COMPONENT_TYPE = Rectangle

def getMaxArea(self):
"""Get area of this block if it were totally full."""
Expand Down
11 changes: 5 additions & 6 deletions armi/reactor/blueprints/componentBlueprint.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,12 @@
"""
import yamlize

from armi import runLog
from armi import materials
from armi.reactor import components
from armi.reactor import composites
from armi import materials, runLog
from armi.nucDirectory import nuclideBases
from armi.reactor import components, composites
from armi.reactor.components import component
from armi.reactor.flags import Flags
from armi.utils import densityTools
from armi.nucDirectory import nuclideBases

COMPONENT_GROUP_SHAPE = "group"

Expand All @@ -42,7 +41,7 @@ def __init__(self, value):
# note: yamlizable does not call an __init__ method, instead it uses __new__ and setattr
self.value = value
if isinstance(value, str):
if not components.COMPONENT_LINK_REGEX.search(value):
if not component.COMPONENT_LINK_REGEX.search(value):
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved
raise ValueError(
"Bad component link `{}`, must be in form `name.dimension`".format(
value
Expand Down
26 changes: 22 additions & 4 deletions armi/reactor/components/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,10 +32,11 @@
import numpy as np

from armi import runLog
from armi.reactor.components.component import * # noqa: F403
from armi.reactor.components.basicShapes import * # noqa: F403
from armi.reactor.components.complexShapes import * # noqa: F403
from armi.reactor.components.volumetricShapes import * # noqa: F403
drewj-tp marked this conversation as resolved.
Show resolved Hide resolved
from armi.reactor.components.component import (
Component,
ComponentType,
componentParameters,
)


def factory(shape, bcomps, kwargs):
Expand Down Expand Up @@ -186,6 +187,23 @@ def getBoundingCircleOuterDiameter(self, Tc=None, cold=False):
"""
return 2 * math.sqrt(self.getComponentArea(cold=cold) / math.pi)

def getCircleInnerDiameter(self, Tc=None, cold=False):
"""
Approximate the component as circular; i.e., inner diameter is zero.

Parameters
----------
Tc : float
Ignored for this component
cold : bool, optional
If True, compute the area with as-input dimensions, instead of thermally-expanded.
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved

Notes
-----
Tc is not used in this method for this particular component.
"""
return 0.0
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems like an okay thing for a shaped component to have to define on their own. Maybe a DerivedShape is the only exception.

Saying this because we try/catch a NotImplementedError below so if Component.getCircleInnerDiameter is required for components, we'd be okay in that context.

But that could be a larger change that is likely better served outside this PR.

Non-blocking comment. If there's traction I can make a ticket

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The assumption for the unshaped component is to return the smallest possible bounding circle outer diameter, i.e., assume the component has a disk shape. It seems consistent with that assumption to assume that the inner diameter is 0.0.

The exception is caught in __lt__, but only for the purpose of producing a clearer error message. The error is still raised and the code crashes. So we still need to be able to sort UnshapedComponents.


@staticmethod
def fromComponent(otherComponent):
"""
Expand Down
2 changes: 1 addition & 1 deletion armi/reactor/components/basicShapes.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,8 +20,8 @@
"""
import math

from armi.reactor.components import ShapedComponent
from armi.reactor.components import componentParameters
from armi.reactor.components.component import ShapedComponent


class Circle(ShapedComponent):
Expand Down
5 changes: 2 additions & 3 deletions armi/reactor/components/complexShapes.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,8 @@

import math

from armi.reactor.components import ShapedComponent
from armi.reactor.components import componentParameters
from armi.reactor.components import basicShapes
from armi.reactor.components import basicShapes, componentParameters
from armi.reactor.components.component import ShapedComponent


class HoledHexagon(basicShapes.Hexagon):
Expand Down
39 changes: 23 additions & 16 deletions armi/reactor/components/component.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,16 +22,11 @@

import numpy as np

from armi import materials
from armi import runLog
from armi import materials, runLog
from armi.bookkeeping import report
from armi.materials import custom
from armi.materials import material
from armi.materials import void
from armi.materials import custom, material, void
from armi.nucDirectory import nuclideBases
from armi.reactor import composites
from armi.reactor import flags
from armi.reactor import parameters
from armi.reactor import composites, flags, parameters
from armi.reactor.components import componentParameters
from armi.utils import densityTools
from armi.utils.units import C_TO_K
Expand Down Expand Up @@ -116,7 +111,7 @@ def __str__(self):

class ComponentType(composites.CompositeModelType):
"""
ComponetType is a metaclass for storing and initializing Component subclass types.
ComponentType is a metaclass for storing and initializing Component subclass types.

The construction of Component subclasses is being done through factories for ease of
user input. As a consequence, the ``__init__`` methods' arguments need to be known
Expand Down Expand Up @@ -278,14 +273,26 @@ def __lt__(self, other):
thisOD = self.getBoundingCircleOuterDiameter(cold=True)
thatOD = other.getBoundingCircleOuterDiameter(cold=True)
try:
return thisOD < thatOD
except Exception:
raise ValueError(
"Components 1 ({} with OD {}) and 2 ({} and OD {}) cannot be ordered because their "
"bounding circle outer diameters are not comparable.".format(
self, thisOD, other, thatOD
if thisOD == thatOD:
thisID = self.getCircleInnerDiameter(cold=True)
thatID = other.getCircleInnerDiameter(cold=True)
return thisID < thatID
else:
return thisOD < thatOD
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved
except (NotImplementedError, Exception) as e:
if isinstance(e, NotImplementedError):
raise NotImplementedError(
"getCircleInnerDiameter not implemented for at least one of {}, {}".format(
self, other
)
)
else:
raise ValueError(
"Components 1 ({} with OD {}) and 2 ({} and OD {}) cannot be ordered because their "
"bounding circle outer diameters are not comparable.".format(
self, thisOD, other, thatOD
)
)
)

def __setstate__(self, state):
composites.Composite.__setstate__(self, state)
Expand Down
12 changes: 10 additions & 2 deletions armi/reactor/components/volumetricShapes.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
import math

from armi.reactor.components import componentParameters
from armi.reactor.components import ShapedComponent
from armi.reactor.components.component import ShapedComponent


class Sphere(ShapedComponent):
Expand Down Expand Up @@ -63,6 +63,10 @@ def getBoundingCircleOuterDiameter(self, Tc=None, cold=False):
"""Abstract bounding circle method that should be overwritten by each shape subclass."""
return self.getDimension("od")

def getCircleInnerDiameter(self, Tc=None, cold=False):
"""Abstract bounding circle method that should be overwritten by each shape subclass."""
return self.getDimension("id")

def getComponentArea(self, cold=False):
"""Compute an average area over the height."""
from armi.reactor.blocks import Block # avoid circular import
Expand Down Expand Up @@ -250,7 +254,11 @@ def getComponentVolume(self):
return vol

def getBoundingCircleOuterDiameter(self, Tc=None, cold=False):
return self.getDimension("outer_radius", Tc, cold)
return 2.0 * self.getDimension("outer_radius", Tc, cold)
mgjarrett marked this conversation as resolved.
Show resolved Hide resolved

def getCircleInnerDiameter(self, Tc=None, cold=False):
"""Abstract bounding circle method that should be overwritten by each shape subclass."""
return 2.0 * self.getDimension("inner_radius")


class DifferentialRadialSegment(RadialSegment):
Expand Down
15 changes: 6 additions & 9 deletions armi/reactor/converters/blockConverters.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@

from armi import runLog
from armi.reactor import blocks, components, grids
from armi.reactor.components.basicShapes import Circle
from armi.reactor.flags import Flags

SIN60 = math.sin(math.radians(60.0))
Expand Down Expand Up @@ -133,7 +134,7 @@ def _checkInputs(self, soluteName, solventName, solute, solvent):
)
if not (
isinstance(solvent, components.DerivedShape)
or all(isinstance(c, components.Circle) for c in (solute, solvent))
or all(isinstance(c, Circle) for c in (solute, solvent))
):
raise ValueError(
"Components are not of compatible shape to be merged "
Expand Down Expand Up @@ -187,11 +188,7 @@ def _verifyExpansion(self, solute, solvent):
c for c in self._sourceBlock if not isinstance(c, components.DerivedShape)
)
for c in sorted(validComponents):
if (
not isinstance(c, components.Circle)
or c is solvent
or c.containsVoidMaterial()
):
if not isinstance(c, Circle) or c is solvent or c.containsVoidMaterial():
continue
if c.isEncapsulatedBy(solvent):
raise ValueError(
Expand Down Expand Up @@ -407,7 +404,7 @@ def _addBlockRings(
assert numFuelBlocksInRing is not None
fuelBlockTotalArea = numFuelBlocksInRing * blockToAdd.getArea()
driverOuterDiam = getOuterDiamFromIDAndArea(innerDiam, fuelBlockTotalArea)
driverRing = components.Circle(
driverRing = Circle(
blockName,
newCompProps,
tempInput,
Expand Down Expand Up @@ -713,7 +710,7 @@ def _buildNonPinRings(self, nonPins):

@staticmethod
def _addSolidMaterialRing(baseComponent, innerDiameter, outDiameter, name):
circle = components.Circle(
circle = Circle(
name,
baseComponent.material,
baseComponent.temperatureInC,
Expand All @@ -729,7 +726,7 @@ def _addSolidMaterialRing(baseComponent, innerDiameter, outDiameter, name):
def _addCoolantRing(self, coolantOD, nameSuffix):
innerDiam = self.convertedBlock[-1].getDimension("od")
irc = self.interRingComponent
interRing = components.Circle(
interRing = Circle(
irc.name + nameSuffix,
irc.material,
irc.temperatureInC,
Expand Down
Loading
Loading