From b3eb91fee691af348653419b402bb19903b3b649 Mon Sep 17 00:00:00 2001 From: Ryan Kingsbury Date: Fri, 26 Jul 2024 09:19:21 -0400 Subject: [PATCH 1/2] Equlibrate: fix bug with pH charge balancing; mypy --- CHANGELOG.md | 4 ++- src/pyEQL/engines.py | 73 +++++++++++++++++++++++--------------------- 2 files changed, 42 insertions(+), 35 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2332d8fc..f518df19 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,10 +5,12 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## Unreleased +## [1.0.4] - 2024-07-26 ### Fixed +- `equilibrate`: Fixed a bug that could cause an `AttributeError` when pH was used for charge + balancing. - Database: `size.radius_ionic` was missing units for `Ni[+2]` and `Cr[+3]`. Correct units have been added. ### Changed diff --git a/src/pyEQL/engines.py b/src/pyEQL/engines.py index bb1bc0b9..824d24c2 100644 --- a/src/pyEQL/engines.py +++ b/src/pyEQL/engines.py @@ -11,7 +11,7 @@ import warnings from abc import ABC, abstractmethod from pathlib import Path -from typing import Literal +from typing import TYPE_CHECKING, Literal from phreeqpython import PhreeqPython @@ -26,6 +26,9 @@ logger = logging.getLogger(__name__) +if TYPE_CHECKING: + from pyEQL import Solution # noqa: TCH004 + class EOS(ABC): """ @@ -38,7 +41,7 @@ class EOS(ABC): """ @abstractmethod - def get_activity_coefficient(self, solution, solute): + def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. @@ -55,7 +58,7 @@ def get_activity_coefficient(self, solution, solute): """ @abstractmethod - def get_osmotic_coefficient(self, solution): + def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of a Solution. @@ -70,7 +73,7 @@ def get_osmotic_coefficient(self, solution): """ @abstractmethod - def get_solute_volume(self): + def get_solute_volume(self, solution: Solution) -> ureg.Quantity: """ Return the volume of only the solutes. @@ -85,7 +88,7 @@ def get_solute_volume(self): """ @abstractmethod - def equilibrate(self, solution): + def equilibrate(self, solution: Solution) -> None: """ Adjust the speciation and pH of a Solution object to achieve chemical equilibrium. @@ -105,25 +108,25 @@ def equilibrate(self, solution): class IdealEOS(EOS): """Ideal solution equation of state engine.""" - def get_activity_coefficient(self, solution, solute): + def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. """ return ureg.Quantity(1, "dimensionless") - def get_osmotic_coefficient(self, solution): + def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of solute, given a Solution object. """ return ureg.Quantity(1, "dimensionless") - def get_solute_volume(self, solution): + def get_solute_volume(self, solution: Solution) -> ureg.Quantity: """Return the volume of the solutes.""" return ureg.Quantity(0, "L") - def equilibrate(self, solution): + def equilibrate(self, solution: Solution) -> None: """Adjust the speciation of a Solution object to achieve chemical equilibrium.""" warnings.warn("equilibrate() has no effect in IdealEOS!") return @@ -138,7 +141,9 @@ class NativeEOS(EOS): def __init__( self, - phreeqc_db: Literal["vitens.dat", "wateq4f_PWN.dat", "pitzer.dat", "llnl.dat", "geothermal.dat"] = "llnl.dat", + phreeqc_db: Literal[ + "phreeqc.dat", "vitens.dat", "wateq4f_PWN.dat", "pitzer.dat", "llnl.dat", "geothermal.dat" + ] = "llnl.dat", ) -> None: """ Args: @@ -172,7 +177,7 @@ def __init__( # store the solution composition to see whether we need to re-instantiate the solution self._stored_comp = None - def _setup_ppsol(self, solution): + def _setup_ppsol(self, solution: Solution) -> None: """Helper method to set up a PhreeqPython solution for subsequent analysis.""" self._stored_comp = solution.components.copy() solv_mass = solution.solvent_mass.to("kg").magnitude @@ -244,13 +249,13 @@ def _setup_ppsol(self, solution): self.ppsol = ppsol - def _destroy_ppsol(self): - """Remove the PhreeqPython solution from memory""" + def _destroy_ppsol(self) -> None: + """Remove the PhreeqPython solution from memory.""" if self.ppsol is not None: self.ppsol.forget() self.ppsol = None - def get_activity_coefficient(self, solution, solute): + def get_activity_coefficient(self, solution: Solution, solute: str): r""" Whenever the appropriate parameters are available, the Pitzer model [may]_ is used. If no Pitzer parameters are available, then the appropriate equations are selected @@ -329,7 +334,7 @@ def get_activity_coefficient(self, solution, solute): # show an error if no salt can be found that contains the solute if salt is None: - logger.error("No salts found that contain solute %s. Returning unit activity coefficient." % solute) + logger.error(f"No salts found that contain solute {solute}. Returning unit activity coefficient.") return ureg.Quantity(1, "dimensionless") # use the Pitzer model for higher ionic strength, if the parameters are available @@ -344,14 +349,14 @@ def get_activity_coefficient(self, solution, solute): # alpha1 and alpha2 based on charge if salt.nu_cation >= 2 and salt.nu_anion <= -2: if salt.nu_cation >= 3 or salt.nu_anion <= -3: - alpha1 = 2 - alpha2 = 50 + alpha1 = 2.0 + alpha2 = 50.0 else: alpha1 = 1.4 alpha2 = 12 else: - alpha1 = 2 - alpha2 = 0 + alpha1 = 2.0 + alpha2 = 0.0 # determine the average molality of the salt # this is necessary for solutions inside e.g. an ion exchange @@ -427,7 +432,7 @@ def get_activity_coefficient(self, solution, solute): return molal - def get_osmotic_coefficient(self, solution): + def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: r""" Return the *molal scale* osmotic coefficient of solute, given a Solution object. @@ -574,7 +579,7 @@ def get_osmotic_coefficient(self, solution): # this means the solution is empty return 1 - def get_solute_volume(self, solution): + def get_solute_volume(self, solution: Solution) -> ureg.Quantity: """Return the volume of the solutes.""" # identify the predominant salt in the solution salt = solution.get_salt() @@ -596,14 +601,14 @@ def get_solute_volume(self, solution): # alpha1 and alpha2 based on charge if salt.nu_cation >= 2 and salt.nu_anion >= 2: if salt.nu_cation >= 3 or salt.nu_anion >= 3: - alpha1 = 2 - alpha2 = 50 + alpha1 = 2.0 + alpha2 = 50.0 else: alpha1 = 1.4 alpha2 = 12 else: - alpha1 = 2 - alpha2 = 0 + alpha1 = 2.0 + alpha2 = 0.0 apparent_vol = ac.get_apparent_volume_pitzer( solution.ionic_strength, @@ -633,7 +638,7 @@ def get_solute_volume(self, solution): pitzer_calc = True - logger.debug("Updated solution volume using Pitzer model for solute %s" % salt.formula) + logger.debug(f"Updated solution volume using Pitzer model for solute {salt.formula}") # add the partial molar volume of any other solutes, except for water # or the parent salt, which is already accounted for by the Pitzer parameters @@ -649,7 +654,7 @@ def get_solute_volume(self, solution): part_vol = solution.get_property(solute, "size.molar_volume") if part_vol is not None: solute_vol += part_vol * ureg.Quantity(mol, "mol") - logger.debug("Updated solution volume using direct partial molar volume for solute %s" % solute) + logger.debug(f"Updated solution volume using direct partial molar volume for solute {solute}") else: logger.warning( @@ -658,7 +663,7 @@ def get_solute_volume(self, solution): return solute_vol.to("L") - def equilibrate(self, solution): + def equilibrate(self, solution: Solution) -> None: """Adjust the speciation of a Solution object to achieve chemical equilibrium.""" if self.ppsol is not None: self.ppsol.forget() @@ -699,7 +704,7 @@ def equilibrate(self, solution): if solution.balance_charge is None: pass elif solution.balance_charge == "pH": - solution.components["H+"] += charge_adjust.magnitude + solution.components["H+"] += charge_adjust elif solution.balance_charge == "pE": raise NotImplementedError else: @@ -734,7 +739,7 @@ class PhreeqcEOS(NativeEOS): def __init__( self, phreeqc_db: Literal[ - "vitens.dat", "wateq4f_PWN.dat", "pitzer.dat", "llnl.dat", "geothermal.dat" + "phreeqc.dat", "vitens.dat", "wateq4f_PWN.dat", "pitzer.dat", "llnl.dat", "geothermal.dat" ] = "phreeqc.dat", ) -> None: """ @@ -751,12 +756,12 @@ def __init__( """ super().__init__(phreeqc_db=phreeqc_db) - def get_activity_coefficient(self, solution, solute): + def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. """ - if self.ppsol is None or solution.components != self._stored_comp: + if (self.ppsol is None) or (solution.components != self._stored_comp): self._destroy_ppsol() self._setup_ppsol(solution) @@ -775,7 +780,7 @@ def get_activity_coefficient(self, solution, solute): return ureg.Quantity(act, "dimensionless") - def get_osmotic_coefficient(self, solution): + def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of solute, given a Solution object. @@ -787,7 +792,7 @@ def get_osmotic_coefficient(self, solution): # TODO - find a way to access or calculate osmotic coefficient return ureg.Quantity(1, "dimensionless") - def get_solute_volume(self, solution): + def get_solute_volume(self, solution: Solution) -> ureg.Quantity: """Return the volume of the solutes.""" # TODO - phreeqc seems to have no concept of volume, but it does calculate density return ureg.Quantity(0, "L") From f7d39e9e5c52b914940ec59eb6af662216f3930d Mon Sep 17 00:00:00 2001 From: Ryan Kingsbury Date: Fri, 26 Jul 2024 09:25:52 -0400 Subject: [PATCH 2/2] type hinting fix --- src/pyEQL/engines.py | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/src/pyEQL/engines.py b/src/pyEQL/engines.py index 824d24c2..4f7a30ea 100644 --- a/src/pyEQL/engines.py +++ b/src/pyEQL/engines.py @@ -27,7 +27,7 @@ logger = logging.getLogger(__name__) if TYPE_CHECKING: - from pyEQL import Solution # noqa: TCH004 + from pyEQL import Solution class EOS(ABC): @@ -41,7 +41,7 @@ class EOS(ABC): """ @abstractmethod - def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: + def get_activity_coefficient(self, solution: "Solution", solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. @@ -58,7 +58,7 @@ def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quan """ @abstractmethod - def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: + def get_osmotic_coefficient(self, solution: "Solution") -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of a Solution. @@ -73,7 +73,7 @@ def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: """ @abstractmethod - def get_solute_volume(self, solution: Solution) -> ureg.Quantity: + def get_solute_volume(self, solution: "Solution") -> ureg.Quantity: """ Return the volume of only the solutes. @@ -88,7 +88,7 @@ def get_solute_volume(self, solution: Solution) -> ureg.Quantity: """ @abstractmethod - def equilibrate(self, solution: Solution) -> None: + def equilibrate(self, solution: "Solution") -> None: """ Adjust the speciation and pH of a Solution object to achieve chemical equilibrium. @@ -108,25 +108,25 @@ def equilibrate(self, solution: Solution) -> None: class IdealEOS(EOS): """Ideal solution equation of state engine.""" - def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: + def get_activity_coefficient(self, solution: "Solution", solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. """ return ureg.Quantity(1, "dimensionless") - def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: + def get_osmotic_coefficient(self, solution: "Solution") -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of solute, given a Solution object. """ return ureg.Quantity(1, "dimensionless") - def get_solute_volume(self, solution: Solution) -> ureg.Quantity: + def get_solute_volume(self, solution: "Solution") -> ureg.Quantity: """Return the volume of the solutes.""" return ureg.Quantity(0, "L") - def equilibrate(self, solution: Solution) -> None: + def equilibrate(self, solution: "Solution") -> None: """Adjust the speciation of a Solution object to achieve chemical equilibrium.""" warnings.warn("equilibrate() has no effect in IdealEOS!") return @@ -177,7 +177,7 @@ def __init__( # store the solution composition to see whether we need to re-instantiate the solution self._stored_comp = None - def _setup_ppsol(self, solution: Solution) -> None: + def _setup_ppsol(self, solution: "Solution") -> None: """Helper method to set up a PhreeqPython solution for subsequent analysis.""" self._stored_comp = solution.components.copy() solv_mass = solution.solvent_mass.to("kg").magnitude @@ -255,7 +255,7 @@ def _destroy_ppsol(self) -> None: self.ppsol.forget() self.ppsol = None - def get_activity_coefficient(self, solution: Solution, solute: str): + def get_activity_coefficient(self, solution: "Solution", solute: str): r""" Whenever the appropriate parameters are available, the Pitzer model [may]_ is used. If no Pitzer parameters are available, then the appropriate equations are selected @@ -432,7 +432,7 @@ def get_activity_coefficient(self, solution: Solution, solute: str): return molal - def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: + def get_osmotic_coefficient(self, solution: "Solution") -> ureg.Quantity: r""" Return the *molal scale* osmotic coefficient of solute, given a Solution object. @@ -579,7 +579,7 @@ def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: # this means the solution is empty return 1 - def get_solute_volume(self, solution: Solution) -> ureg.Quantity: + def get_solute_volume(self, solution: "Solution") -> ureg.Quantity: """Return the volume of the solutes.""" # identify the predominant salt in the solution salt = solution.get_salt() @@ -663,7 +663,7 @@ def get_solute_volume(self, solution: Solution) -> ureg.Quantity: return solute_vol.to("L") - def equilibrate(self, solution: Solution) -> None: + def equilibrate(self, solution: "Solution") -> None: """Adjust the speciation of a Solution object to achieve chemical equilibrium.""" if self.ppsol is not None: self.ppsol.forget() @@ -756,7 +756,7 @@ def __init__( """ super().__init__(phreeqc_db=phreeqc_db) - def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quantity: + def get_activity_coefficient(self, solution: "Solution", solute: str) -> ureg.Quantity: """ Return the *molal scale* activity coefficient of solute, given a Solution object. @@ -780,7 +780,7 @@ def get_activity_coefficient(self, solution: Solution, solute: str) -> ureg.Quan return ureg.Quantity(act, "dimensionless") - def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: + def get_osmotic_coefficient(self, solution: "Solution") -> ureg.Quantity: """ Return the *molal scale* osmotic coefficient of solute, given a Solution object. @@ -792,7 +792,7 @@ def get_osmotic_coefficient(self, solution: Solution) -> ureg.Quantity: # TODO - find a way to access or calculate osmotic coefficient return ureg.Quantity(1, "dimensionless") - def get_solute_volume(self, solution: Solution) -> ureg.Quantity: + def get_solute_volume(self, solution: "Solution") -> ureg.Quantity: """Return the volume of the solutes.""" # TODO - phreeqc seems to have no concept of volume, but it does calculate density return ureg.Quantity(0, "L")