-
Notifications
You must be signed in to change notification settings - Fork 53
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
Added ase Nudged Elastic Band in the NewtonNet recipe #2176
Conversation
Can one of the admins verify this patch? |
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #2176 +/- ##
==========================================
+ Coverage 97.44% 97.46% +0.02%
==========================================
Files 85 86 +1
Lines 3557 3708 +151
==========================================
+ Hits 3466 3614 +148
- Misses 91 94 +3 ☔ View full report in Codecov by Sentry. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @kumaranu: I know this is still a WIP but I wanted to provide some initial comments. The biggest thing to note is that in quacc you cannot write things out to the current working directory or any relative paths. Quacc does not cd
into the working directory and so there is no guarantee that the current working directory or any relative path is going to be where the runtime/results directory is.
If you are looking to write out files before the calculation completes, then you need write it out based on the tmpdir
that is made when quacc.runners.pre.calc_setup
is called. For instance, in phonopy, you can see that certain files are written out (e.g. phonopy.yaml
, phonopy_auto_band_structure.yaml
) based on the tmpdir
of the runtime directory:
quacc/src/quacc/runners/phonons.py
Lines 55 to 68 in 04af31e
# Perform staging operations | |
tmpdir, job_results_dir = calc_setup(atoms) | |
# Run phonopy | |
phonon.forces = forces | |
phonon.produce_force_constants() | |
phonon.run_mesh(with_eigenvectors=True) | |
phonon.run_total_dos() | |
phonon.run_thermal_properties(t_step=t_step, t_max=t_max, t_min=t_min) | |
phonon.auto_band_structure( | |
write_yaml=True, filename=Path(tmpdir, "phonopy_auto_band_structure.yaml") | |
) | |
phonon.save(Path(tmpdir, "phonopy.yaml"), settings={"force_constants": True}) | |
phonon.directory = job_results_dir |
If you are writing to disk after the calculation completes and a schema is returned, you can query schema["dir_name"]
to get the path where the results folder is. In any case, there is going to need to be a fairly significant amount of refactoring here to ensure that your I/O goes where you expect it to go. These functions may have been working fine for you because you are using FireWorks which cd
's into a custom directory for each run, but that is very unique to FireWorks. If you were to run these calculations without a workflow engine, for instance, you'll see that the files are not being written out to the unique results folder for that calculation.
Most of your issues in this PR boil down to the fact that you are trying to put too much into the recipes themselves, which are meant to be very well-defined units of work. In reality, most of your additions here should be quacc.runners
functions that are called within your new recipes. I/O happens in runners and schemas depending on if it's meant to be done before/during or after the calculation. If using ChatGPT, it has no knowledge of this.
pyproject.toml
Outdated
@@ -49,7 +49,7 @@ defects = ["pymatgen-analysis-defects>=2023.8.22", "shakenbreak>=3.2.0"] | |||
jobflow = ["jobflow[fireworks]>=0.1.14", "jobflow-remote>=0.1.0"] | |||
mlp = ["matgl>=1.0.0", "chgnet>=0.3.3", "mace-torch>=0.3.3", "torch-dftd>=0.4.0"] | |||
mp = ["atomate2>=0.0.14"] | |||
newtonnet = ["newtonnet>=1.1"] | |||
newtonnet = ["newtonnet>=1.1", "geodesic-interpolate @ git+https://github.com/virtualzx-nad/geodesic-interpolate.git"] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We can't have direct git links in pyproject.toml
(unfortunately) because PyPI does not allow us to upload new version of quacc with URLs. The best course of action here is to simply make it clear in the @requires
decorator that this package should be installed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @Andrew-S-Rosen ,
I want to have tests that run for geodesic interpolate on GitHub. Is there a way you see to do that in case we are not able to put URLs?
Thanks.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@kumaranu: Simply remove the direct link from pyproject.toml
. The requirements.txt
files in the tests
folder are what is used by CI.
I have added a function in runners/ase.py with its test. My test is running alright locally but it is complaining that the required packages are not installed such as "sella", "NewtonNet", "geodesic" etc. Maybe the reason for this problem is that requirements-newtonnet.txt is not called for the packages that are present inside runners/ase.py. But I do not know which file will used to get that list of required packages. Please help resolve this issue. Thanks. Tagging Sam here as I was discussing this issue with him recently. |
I added the following three lines to requirements.txt inside the tests directory to solve that problem: |
@@ -133,7 +133,7 @@ def test_ase_relax_job(tmp_path, monkeypatch): | |||
output["parameters"]["orcasimpleinput"] | |||
== "def2-tzvp engrad normalprint wb97x-d3bj xyzfile" | |||
) | |||
assert output["parameters_opt"]["fmax"] == 0.1 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please don't change pre-existing tests. They should pass just fine.
pyproject.toml
Outdated
@@ -50,7 +50,7 @@ jobflow = ["jobflow[fireworks]>=0.1.14", "jobflow-remote>=0.1.0"] | |||
mlp = ["matgl>=1.1.2", "chgnet>=0.3.3", "mace-torch>=0.3.3", "torch-dftd>=0.4.0"] | |||
mp = ["atomate2>=0.0.14"] | |||
newtonnet = ["newtonnet>=1.1"] | |||
parsl = ["parsl[monitoring]>=2024.5.27; platform_system!='Windows'"] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please don't change unrelated versions. They should work just fine.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't know what was changed here. I see that this file is the same as the one in Quantum-Accelerator: main. Maybe you changed it back already.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Did I change the parsl version from 2023.10.23 to 2.24.5.27?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, I took care of it. However, there are several other changes that still need to be reverted back, as shown in the file comparison view.
Although the original problem is solved, I saw two new errors, tblite and mlp tests both of which I did not touch. mlp test were just complaining that they could not find mace. I added the following lines in the requirements.txt to get rid of that error. However, the error in tblite seems much different as the tests ran fine but the numbers don't match. I have left that as it is: |
@kumaranu: The I do not quite understand what your goal is here with |
I have removed NewtonNet and Sella from the test_ase.py requirements inside runners directory. It is using BFGS and EMT now. |
@kumaranu: Thanks. I will review the PR once you undo all the unrelated changes you've made (there are still many) and remove commented out code blocks. |
src/quacc/runners/ase.py
Outdated
@@ -320,6 +326,153 @@ def run_vib( | |||
return vib | |||
|
|||
|
|||
def run_path_opt( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's rename this run_neb
.
src/quacc/schemas/ase.py
Outdated
@@ -275,6 +275,40 @@ def summarize_vib_and_thermo( | |||
) | |||
|
|||
|
|||
def summarize_path_opt_run(dyn: Optimizer) -> OptSchema: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
And then we can rename this summarize_neb_run
. It also needs a proper docstring.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some more changes to revert later.
tests/core/schemas/test_ase.py
Outdated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some more changes to revert later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @kumaranu. I have provided an initial review below. There are many areas for cleanup, but overall the logic seems mostly reasonable. If you can address these changes, that would be ideal. Please note that you should also be checking GitHub Actions CI to see what tests fail. You'll see that your docs and pre-commit CI runners failed. Clicking the "details" will show you why.
Once you make these changes, I will review it once more and can assist with remaining issues. Thank you. Please also "resolve" the comments when you are done addressing them.
tests/core/runners/test_ase.py
Outdated
) | ||
], | ||
) | ||
def test_run_neb( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Your test of run_neb
should not rely on geodesic-interpolate
. We want to make sure that we are only testing one thing at a time (hence the name a "unit test").
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The NEB function needs list of ASE images. I am avoiding it because it will just become a long list of coordinates in this file. Not sure how to do it cleanly.
tests/core/runners/test_ase.py
Outdated
assert neb_summary["trajectory_results"][1]["energy"] == pytest.approx( | ||
1.098, abs=0.01 | ||
) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A more thorough test of the output schema is required. Also, in general, let's not use such a loose tolerance if possible. This will help us capture errors appropriately.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same problem as the geodesic here because I am using geodesic as the precursor to this step.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are too many things being tested here at once is the major problem. For instance, runners/test_ase.py
should be testing the runner: run_neb
. It should be asserting that run_neb()
functions appropriately and nothing else. There should be no major precursor to that step and nothing afterwards (i.e. no summarizing).
You'd then make a separate test for summarizing (e.g. tests.cores.schemas
) and so on and so forth.
The idea of a "unit test" is that you are testing a specific unit of work.
tests/core/schemas/test_ase.py
Outdated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please do not make unrelated changes in your PR like the code changes in this file.
Hi @Andrew-S-Rosen, I am not sure why one of my tests is failing on Windows all of a sudden. Please let me know if you see anything obvious to you. Thanks. |
It looks like a tolerance issue and should be not a problem. You can change the tolerance slightly (just enough for the tests to pass). I am not sure the source either. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
src/quacc/recipes/newtonnet/ts.py
Outdated
class NebSchema(TypedDict): | ||
relax_reactant: OptSchema | ||
relax_product: OptSchema | ||
geodesic_results: list[Atoms] | ||
neb_results: dict | ||
|
||
class NebTsSchema(TypedDict): | ||
relax_reactant: OptSchema | ||
relax_product: OptSchema | ||
geodesic_results: list[Atoms] | ||
neb_results: dict | ||
ts_results: TSSchema |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The neb_results: dict
should use the type hint for the return from summarize_neb_run
(if we keep that function) rather than dict
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not sure how to do this one. I see that quacc.types has the base classes to define the schemas but I don't have those for summarize_neb_run.
src/quacc/recipes/newtonnet/ts.py
Outdated
@job | ||
@requires( | ||
has_newtonnet, "NewtonNet must be installed. Refer to the quacc documentation." | ||
) | ||
@requires( | ||
has_geodesic_interpolate, | ||
"geodesic-interpolate must be installed. Refer to the quacc documentation.", | ||
) | ||
def neb_job( | ||
reactant_atoms: Atoms, | ||
product_atoms: Atoms, | ||
relax_job_kwargs: dict[str, Any] | None = None, | ||
calc_kwargs: dict[str, Any] | None = None, | ||
geodesic_interpolate_kwargs: dict[str, Any] | None = None, | ||
neb_kwargs: dict[str, Any] | None = None, | ||
) -> NebSchema: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I haven't commented on this previously but really should now:
In quacc, the @job
is a discrete unit of work. A @flow
is a workflow, concatenating multiple distinct @job
s. Here (and in your other recipes, for what it's worth), you are actually doing a workflow. You have two relaxations (two "jobs") before running an NEB calculation.
Normally, I would say this is not permitted. If we were running a Q-Chem calculation for instance, this would be highly undesirable because you would want the reactant and product relaxation jobs to be done concurrently, you'd want those results to be stored in your database, and if one fails you'd still like to store the results from the other. You can't do that if everything is packed in one @job
.
The challenge here is that because NewtonNet and MLPs are so fast, it is tempting to pack everything into one @job
. For users of workflow engines (like FireWorks) where each @job
is a Slurm job, this means you'd be queuing more than you need. But not all workflow engines work that way (and even FireWorks has the option to not re-queue each job).
Jobflow does not support the @flow
pattern, so I'm not going to inconvenience you by requiring that you implement it here. But I wanted to bring up this difference: what you are actually doing here is making a flow that combines steps from multiple jobs.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the recipes like neb_ts_job and geodesic_ts_job will most likely go away. I have left them working right now but will probably remove them after a discussion together. I have put the workflows on a separate repository so things like the flow can be done over there. I have however not use jobflow over there so the parallel nature of the execution of the codes and also the fireworks related functionalities don't exist in those workflows as well.
# Conflicts: # src/quacc/runners/ase.py
# Conflicts: # src/quacc/runners/ase.py
Hi @samblau , I reduced the timing of the tests by adding max_iter to them and also got rid of writing to the current directory problem. Let's see if it works out OK or not. Thanks! |
src/quacc/schemas/ase.py
Outdated
if interval > n_iter: | ||
pass |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@kumaranu: FYI that this code block doesn't actually do anything. I'm going to remove it, but if you meant for it to do something else just let me know.
Summary of Changes
>>
Target here is to add ase's NEB as the double ended transition state search method for NewtonNet recipe.
<<
Checklist
main
.Notes