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

Move cold height expansion to bp #1047

Merged
merged 12 commits into from
Dec 29, 2022
Merged

Move cold height expansion to bp #1047

merged 12 commits into from
Dec 29, 2022

Conversation

onufer
Copy link
Member

@onufer onufer commented Dec 22, 2022

Description


Checklist

  • This PR has only one purpose or idea.
  • Tests have been added/updated to verify that the new/changed code works.
  • The release notes (location doc/release/0.X.rst) are up-to-date with any bug fixes or new features.
  • The documentation is still up-to-date in the doc folder.
  • The dependencies are still up-to-date in setup.py.

@onufer onufer requested review from albeanth, ntouran and jakehader and removed request for albeanth and ntouran December 22, 2022 01:21
Copy link
Member

@albeanth albeanth left a comment

Choose a reason for hiding this comment

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

I see what you're getting at here. It makes sense. Let's do the expansion on the assemblies at the blueprints level as opposed to every single assembly in the core. For both standard ARMI runs and snapshot/restart runs with loadState, this will be faster and have the same outcome.

With some testing/cleanup etc I think it's a good idea. As part of the PR I think it would be cool to quantify the speedup too.

@onufer
Copy link
Member Author

onufer commented Dec 28, 2022

Load state will be the same since we use what’s in the db for that and only expand blueprints assems with load state (both before and after)

but o=armi.init will be much faster

@onufer onufer requested a review from john-science December 28, 2022 00:15
@onufer
Copy link
Member Author

onufer commented Dec 28, 2022

The new implementation should have the exact same output/methodology as the old, but the implementation is simpler and faster, which is nice.

@@ -102,6 +104,7 @@
from armi.reactor.blueprints.componentBlueprint import ComponentGroups
from armi.reactor.blueprints import isotopicOptions
from armi.reactor.blueprints.gridBlueprint import Grids, Triplet
from armi.reactor.converters.axialExpansionChanger import AxialExpansionChanger
Copy link
Member

Choose a reason for hiding this comment

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

Hmmm

So, the primary blueprint code knows about the Axial Expansion?

But two-thirds of the projects that use ARMI, and half the ones at TerraPower, don't even have axial assemblies.

So, how does this code work for, say, a molten chloride reactor?

I ask for two reasons:

  1. ARMI has to work for all reactors, not just one.
  2. Axial Expansion is very slow, and for sure we don't want to instantiate all that unless we have to.

Copy link
Member Author

@onufer onufer Dec 28, 2022

Choose a reason for hiding this comment

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

If you look at the code it’s only called if input heights considered cold

Both before and after this change the same calls are made by the time o=armi.init is done running. o=armi.init is about the minimal thing you can do and the new code will be just as fast or faster for those circumstances (because its producing the correct assemblies before the deepcopy, rather than deepcopying and then fixing them all after....

Copy link
Member

Choose a reason for hiding this comment

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

That's a fair point. Okay, so we aren't breaking non-axial-type reactors.

But we are importing something that is specific to a TYPE of reactor (albeit a common one) into a part of the code that (right now) reads YAML files and builds some math.

It just feels like something that should be in assemblyBlueprint.py. But, I assume, it was just hard to work into that module?

Copy link
Member Author

Choose a reason for hiding this comment

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

you need to know the meshes of each of the individual assemblies to go through the processes since they all have their indices mapped to the reference assembly (see a.makeAxialSnapList(referenceAssembly)). I dont see anywhere in that module where there are the full set of assemblies. The nice thing about doing it here is that the blueprints assemblies will exists for the shortest possible time ad the incorrect dimensions/masses since they are fixed right after construction (line 322)

Copy link
Member Author

Choose a reason for hiding this comment

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

if you want, right after the area consistancy check i could move all the new code to
axialExpansionChanger

and have it be

if not cs["detailedAxialExpansion"]:
    axialExpansionChanger.makeAssembliesSnapToUniformMesh(self.assemblies.values(), cs["nonUniformAssemFlags"])
if not cs["inputHeightsConsideredHot"]:
    axialExpansionChanger.expandColdDimsToHot(self.assemblies.values(),  cs["detailedAxialExpansion"])

Fundamentally though, blueprints should be responsible for producing assemblies with correct masses and elevations. It can certainly call someone else to do the work though

Copy link
Member

@john-science john-science Dec 29, 2022

Choose a reason for hiding this comment

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

  1. I see the problem you're trying to solve.
  2. I love that you moved the axial expansion logic into that module.
  3. Taking a step back, I think blueprints/__init__.py already has many direct references to assemblies at the top level. So I think you're not causing a design flaw so much as stuck within the system we created.
  4. Let's explore your solution. The worst case is that each special type of reactor needs a bunch of these if cs["settingName"] statements, which point to some non-blueprint method. Unless that list gets long, that's probably going to look a little ugly, but not be heart-breakingly bad.
  5. Sure, after moving the logic to the axial expansion module, I'm convinced.

Sorry, I'm not trying to be a bother. I have just learned by hard experience that improving something at the start can save a TON of time spent on technical debt later. I keep meaning to give a talk on that.

Copy link
Member Author

@onufer onufer Dec 29, 2022

Choose a reason for hiding this comment

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

Great. thanks for clarifying @john-science. I genuinely don't think this list of things will grow considerably... I would recommend describing molten salt reactors and TRISO-based fuel as "non-pin-based" or "non-assembly-based" as opposed to "non-axial-type reactors".

Molten salt reactors still typically care about axial expansion of their vessel and the impact of that on reactivity (I spent a couple years on a molten salt project). Its just that the framework currently doesn't support it. Its not a trivial feature though and its a bit different from a fuel pin expanding. It took a couple hundred hours for us to get cold to hot dims working for pin-based-reactors. I am sure that if we had the capability to input cold vessel dims and have the framework expand the vessel to hot, molten salt reactors would want to use the capability. Its just we don't have it now, so those folks tend to continue to use hot axial heights.

If/when we have this capability, I would image we would change the call to expandColdDimsToHot to determine a different changer to use for non-pin based reactors rather than having another if statement here in blueprints.

Sure, after moving the logic to the axial expansion module, I'm convinced. what more do you need to approve the PR?

Copy link
Member

Choose a reason for hiding this comment

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

I would recommend describing molten salt reactors and TRISO-based fuel as "non-pin-based" or "non-assembly-based" as opposed to "non-axial-type reactors".

Thanks. And, always, I am interested in learning more about reactor design/analysis. Sorry, my graduate degrees are in physics, but not the right kind. I'm happy to learn.

what more do you need to approve the PR?

I added one little comment about a docstring, but I'm good otherwise. The logic looks sound, and I like the new layout better.

Thanks, man!

@john-science
Copy link
Member

I'm not convinced the axial expansion logic belongs in the blueprints.

What about non-axial reactors? Tony and I worked over a couple of pull requests to pull all references to axial expansion OUT of the blueprints, and just have references to the logic.

What is the benefit of having our default blueprint logic aware of axial expansion? (Please note that two thirds of the projects most heavily using ARMI don't have axial layouts or assemblies. And even at TerraPower, one of our two reactors won't ever need to use ANY axial expansion logic.

Thoughts?

Comment on lines 330 to 351
# if assemblies are defined in blueprints, handle meshing
# assume finest mesh is reference
assemsByNumBlocks = sorted(
self.assemblies.values(),
key=lambda a: len(a),
reverse=True,
)
referenceAssembly = assemsByNumBlocks[0] if assemsByNumBlocks else None

if not cs["detailedAxialExpansion"]:
# make the snap lists so assems know how to expand
nonUniformAssems = [
Flags.fromStringIgnoreErrors(t) for t in cs["nonUniformAssemFlags"]
]
# prepare core for mesh snapping during axial expansion
for a in self.assemblies.values():
if any(a.hasFlags(f) for f in nonUniformAssems):
continue
a.makeAxialSnapList(referenceAssembly)
if not cs["inputHeightsConsideredHot"]:
# expand axial heights from cold to hot
self._coldDimsToHot(referenceAssembly, cs["detailedAxialExpansion"])
Copy link
Member Author

Choose a reason for hiding this comment

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

@john-science happy to move this stuff else where and have a 1 line call to hit here.

@onufer
Copy link
Member Author

onufer commented Dec 28, 2022

@john-science I absolutely think calls should to happen here rather than processCoreLoading. What's currently happening in processCoreLoading is

  1. if its a db load just expand the blueprints since heights are params and correct
  2. if its a first initialization expand both blueprints and assemblies in the reactor. BUT the only reason we have to expand the assems in the reactor is we failed to expand the blueprints before we deep-copied them to populate the reactor. Its a tremendous waste to deepcopy a whole reactor of assemblies that have incorrect masses and axial heights. The reason that we got here is there was a bit of debate about what he blueprints should store. what we landed on is that it should store axially expanded assemblies and it should be the same at each statepoint since its not in the database. therefore doing it here is the least expensive/most clear.

I think most of the logic is out of the blueprints... right now we are just checking some settings and calling outside methods/functions that update the state, but i would be happy to move the remainder out as well and have it be a 1 line call as shown in the comment..

Where would you want this stuff to live?

@onufer
Copy link
Member Author

onufer commented Dec 28, 2022

I do want the call to be front this location though so that the assemblies in blueprints exist at the wrong dimensions and densities as short a time as possible.

@onufer
Copy link
Member Author

onufer commented Dec 29, 2022

@john-science, I moved as much of the logic as possible out of blueprints, and added comments to hopefully communicate better what is happening and why its needed useful.

If you have more comments could you please be specific about what you want done? I feel like i am guessing/stabbing in the dark here... I could have a better chance of addressing the concern adequately if the comments were a little more specific/actionable. Thanks!

Copy link
Member

@john-science john-science left a comment

Choose a reason for hiding this comment

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

Great.

This solves a problem, and does it more by reorganizing already-tested code, than building things from scratch. The testing seems fine, and I love that this PR has one goal, and not 12.

Thanks for that.

@onufer onufer removed the request for review from jakehader December 29, 2022 21:51
@onufer onufer merged commit 94e2500 into main Dec 29, 2022
@onufer
Copy link
Member Author

onufer commented Dec 29, 2022

@albeanth here is what time looks like on small test reactor....
before PR
no expand = 24.5 sec
expand cold to hot = 49.5 sec

after PR
no expand = 21.2 sec
expand cold to hot = 25.9 sec

some reactors take more like 5 minutes to load, so folks should notice a difference.

import armi

armi.configure()

from armi.reactor.tests import test_reactors
import timeit

from armi import context

context.Mode.setMode(context.Mode.BATCH)  # disable yes/no CLI

hot = timeit.timeit(
    'test_reactors.loadTestReactor(customSettings={"inputHeightsConsideredHot": True})',
    setup="from armi.reactor.tests import test_reactors",
    number=10,
)

cold = timeit.timeit(
    'test_reactors.loadTestReactor(customSettings={"inputHeightsConsideredHot": False})',
    setup="from armi.reactor.tests import test_reactors",
    number=10,
)

print(hot, cold)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants