Source code for prommis.uky.uky_flowsheet

#####################################################################################################
# “PrOMMiS” was produced under the DOE Process Optimization and Modeling for Minerals Sustainability
# (“PrOMMiS”) initiative, and is copyright (c) 2023-2024 by the software owners: The Regents of the
# University of California, through Lawrence Berkeley National Laboratory, et al. All rights reserved.
# Please see the files COPYRIGHT.md and LICENSE.md for full copyright and license information.
#####################################################################################################
r"""
University of Kentucky REE Processing Plant
===========================================

Author: Marcus Holly

The University of Kentucky (UKy) rare earth element (REE) processing plant is designed to extract salable rare earth oxides
from domestic U.S. coal and coal byproducts. While this implementation of the plant does not take into account
all of the complexities and unit processes detailed in the 2019 report, it depicts the major phenomena
by utilizing a series of conventional REE extraction techniques,
including acid tank leaching, solvent extraction, precipitation, and product roasting.

Implementation
--------------

Figure 1 shows the process flow diagram for the simplified, steady-state UKy plant where the solid and liquid feeds are
sent to a leaching tank for initial processing; then the solids (with some retained liquid) are separated out as a filter
cake while the remaining liquid is sent to the solvent extraction process which is comprised of 2 distinct circuits.
In the rougher circuit, solutes in the aqueous phase are transferred to the organic phase, and a portion of the
depleted aqueous solution is recycled back to the leaching process while the remainder is sent to the cleaner circuit.
The degree to which components are transferred from one phase to the other is dependent upon the unit's partition coefficient for that particular component.
In the cleaner circuit, solutes in the organic phase are transferred to the aqueous phase, and a portion of the loaded
aqueous solution is recycled back to the rougher circuit while the remainder is sent to the precipitator. The precipitate
(with some retained liquid) is sent to the roaster where the product rare earth oxides are generated, and the liquid from
the precipitator is recycled back to the cleaner circuit.

.. figure:: ../tutorials/uky_flowsheet.png
    :width: 800
    :align: center

    University of Kentucky flowsheet

Degrees of Freedom
------------------
The following variables must be specified by the user to run the UKy flowsheet:
    * liquid feed volumetric flow rate and component concentrations
    * solid feed mass flow and component mass fractions
    * volume of leach tank(s)
    * partition coefficients for each solvent extraction unit
    * flow rate and component concentrations for organic make-up streams and HCl feeds
    * liquid recovery fraction for solid-liquid separators
    * precipitator inlet temperature
    * roaster inlet temperature and pressure, outlet temperature, and pressure drop
    * roaster gas and moisture molar flows, vapor component mole fractions, and oxide recovery fraction
    * split fractions for each recycle loop

Default Flowsheet Specifications
--------------------------------

===================================================================== ============ ============================
Description                                                           Value        Units
===================================================================== ============ ============================
Leaching
Tank volume                                                           100          :math:`\text{m}^3`
Liquid feed volumetric flow                                           224.3        :math:`\text{L/hr}`
Liquid feed H concentration                                           100          :math:`\text{mg/L}`
Liquid feed HSO4 concentration                                        1e-8         :math:`\text{mg/L}`
Liquid feed SO4 concentration                                         4800         :math:`\text{mg/L}`
Liquid feed REE and contaminant concentrations                        1e-10        :math:`\text{mg/L}`
Solid feed mass flow                                                  22.68        :math:`\text{kg/hr}`
Solid feed inerts mass fraction                                       0.6952       :math:`\text{dimensionless}`
Solid feed Al2O3 mass fraction                                        0.237        :math:`\text{dimensionless}`
Solid feed Fe2O3 mass fraction                                        0.0642       :math:`\text{dimensionless}`
Solid feed CaO mass fraction                                          0.00331      :math:`\text{dimensionless}`
Solid feed Sc2O3 mass fraction                                        2.8e-5       :math:`\text{dimensionless}`
Solid feed Y2O3 mass fraction                                         3.3e-5       :math:`\text{dimensionless}`
Solid feed La2O3 mass fraction                                        6.8e-5       :math:`\text{dimensionless}`
Solid feed Ce2O3 mass fraction                                        1.6e-4       :math:`\text{dimensionless}`
Solid feed Pr2O3 mass fraction                                        1.7e-5       :math:`\text{dimensionless}`
Solid feed Nd2O3 mass fraction                                        6.8e-5       :math:`\text{dimensionless}`
Solid feed Sm2O3 mass fraction                                        1.5e-5       :math:`\text{dimensionless}`
Solid feed Gd2O3 mass fraction                                        1.0e-5       :math:`\text{dimensionless}`
Solid feed Dy2O3 mass fraction                                        7.5e-6       :math:`\text{dimensionless}`

Solvent Extraction Rougher
Loading section organic feed volumetric flow                          62.01        :math:`\text{L/hr}`
Organic make-up REE and contaminant concentrations                    1e-7         :math:`\text{mg/L}`
Scrubbing section acid feed volumetric flow                           0.09         :math:`\text{L/hr}`
Scrubbing section acid feed H concentration                           10.36        :math:`\text{mg/L}`
Scrubbing section acid feed Cl concentration                          359.64       :math:`\text{mg/L}`
Scrubbing section acid feed REE and contaminant concentrations        1e-7         :math:`\text{mg/L}`
Stripping section acid feed volumetric flow                           0.09         :math:`\text{L/hr}`
Stripping section acid feed H concentration                           41.44        :math:`\text{mg/L}`
Stripping section acid feed Cl concentration                          1438.56      :math:`\text{mg/L}`
Stripping section acid feed REE and contaminant concentrations        1e-7         :math:`\text{mg/L}`

Solvent Extraction Cleaner
Loading section organic feed volumetric flow                          62.01        :math:`\text{L/hr}`
Organic make-up REE and contaminant concentrations                    1e-7         :math:`\text{mg/L}`
Stripping section acid feed volumetric flow                           0.09         :math:`\text{L/hr}`
Stripping section acid feed H concentration                           41.44        :math:`\text{mg/L}`
Stripping section acid feed Cl concentration                          1438.56      :math:`\text{mg/L}`
Stripping section acid feed REE and contaminant concentrations        1e-7         :math:`\text{mg/L}`

Precipitator
Inlet temperature                                                     348.15       :math:`\text{K}`

Roaster
Pressure drop                                                         0            :math:`\text{Pa}`
Gas inlet temperature                                                 348.15       :math:`\text{K}`
Gas outlet temperature                                                873.15       :math:`\text{K}`
Gas inlet pressure                                                    101325       :math:`\text{Pa}`
Gas inlet molar flow                                                  0.00781      :math:`\text{mol/s}`
Gas inlet O2 mole fraction                                            0.1118       :math:`\text{dimensionless}`
Gas inlet H2O mole fraction                                           0.1005       :math:`\text{dimensionless}`
Gas inlet CO2 mole fraction                                           0.0431       :math:`\text{dimensionless}`
Gas inlet N2 mole fraction                                            0.7446       :math:`\text{dimensionless}`
Moisture inlet molar flow                                             6.75e-4      :math:`\text{mol/s}`
Oxide recovery fraction                                               0.95         :math:`\text{dimensionless}`

Separators
Leaching solid-liquid separator liquid recovery fraction              0.7          :math:`\text{dimensionless}`
Solvent extraction rougher load recycle split fraction                0.9          :math:`\text{dimensionless}`
Solvent extraction rougher scrub recycle split fraction               0.9          :math:`\text{dimensionless}`
Solvent extraction rougher organic recycle split fraction             0.9          :math:`\text{dimensionless}`
Solvent extraction cleaner organic recycle split fraction             0.9          :math:`\text{dimensionless}`
Precipitator solid-liquid separator liquid recovery fraction          0.7          :math:`\text{dimensionless}`
Precipitator solid-liquid separator liquid recycle split fraction     0.9          :math:`\text{dimensionless}`
===================================================================== ============ ============================

Costing
-------
Unit model costing in this flowsheet is preliminary and is based on the commercial scale unit model parameters provided in Table 4-28 :math:`^1`.
However, this flowsheet is at the pilot scale, so while some of the unit model costing parameters have been scaled down
accordingly, a more robust scale-down procedure of the costing parameters is necessary to accurately approximate the cost of this pilot scale system.


References:

[1] Steven Keim, "Production of salable rare earths products from coal and coal byproducts in the U.S.
using advanced separation processes", 2019

"""

from pyomo.environ import (
    check_optimal_termination,
    ConcreteModel,
    Constraint,
    Expression,
    Param,
    SolverFactory,
    Suffix,
    TransformationFactory,
    Var,
    value,
    units,
)
from pyomo.network import Arc, SequentialDecomposition

from idaes.core import (
    FlowDirection,
    FlowsheetBlock,
    MaterialBalanceType,
    MomentumBalanceType,
    UnitModelBlock,
    UnitModelCostingBlock,
)
from idaes.core.solvers import get_solver
from idaes.core.initialization import BlockTriangularizationInitializer
from idaes.core.util.model_statistics import degrees_of_freedom
from idaes.models.properties.modular_properties.base.generic_property import (
    GenericParameterBlock,
)
from idaes.models.unit_models.feed import Feed, FeedInitializer
from idaes.models.unit_models.mixer import (
    Mixer,
    MixingType,
    MomentumMixingType,
    MixerInitializer,
)
from idaes.models.unit_models.product import Product, ProductInitializer
from idaes.models.unit_models.separator import (
    EnergySplittingType,
    Separator,
    SplittingType,
    SeparatorInitializer,
)
from idaes.models.unit_models.solid_liquid import SLSeparator
from idaes.models_extra.power_generation.properties.natural_gas_PR import (
    EosType,
    get_prop,
)
import idaes.logger as idaeslog

from prommis.leaching.leach_train import LeachingTrain
from prommis.leaching.leach_reactions import CoalRefuseLeachingReactions
from prommis.leaching.leach_solids_properties import CoalRefuseParameters
from prommis.leaching.leach_solution_properties import LeachSolutionParameters
from prommis.precipitate.precipitate_liquid_properties import AqueousParameter
from prommis.precipitate.precipitate_solids_properties import PrecipitateParameters
from prommis.precipitate.precipitator import Precipitator
from prommis.roasting.ree_oxalate_roaster import REEOxalateRoaster
from prommis.solvent_extraction.ree_og_distribution import REESolExOgParameters
from prommis.solvent_extraction.solvent_extraction import (
    SolventExtraction,
    SolventExtractionInitializer,
)
from prommis.uky.costing.ree_plant_capcost import QGESSCosting, QGESSCostingData

_log = idaeslog.getLogger(__name__)

# Epsilon represents near-zero component concentrations
eps = 1e-8 * units.mg / units.L


[docs] def main(): """ Run the flowsheet by calling the appropriate functions in series. """ m = build() set_partition_coefficients(m) set_operating_conditions(m) set_scaling(m) scaling = TransformationFactory("core.scale_model") scaled_model = scaling.create_using(m, rename=False) if degrees_of_freedom(scaled_model) != 0: raise AssertionError( "The degrees of freedom are not equal to 0." "Check that the expected variables are fixed and unfixed." "For more guidance, run assert_no_structural_warnings from the IDAES DiagnosticToolbox " ) initialize_system(scaled_model) solve_system(scaled_model) # fixes the volumetric flow rate of the organic recycle streams and unfixes the flow of the make-up streams # we want to be able to adjust the total recycle flow rate, not just the make-up portion of it fix_organic_recycle(scaled_model) scaled_results = solve_system(scaled_model) if not check_optimal_termination(scaled_results): raise RuntimeError( "Solver failed to terminate with an optimal solution. Please check the solver logs for more details" ) results = scaling.propagate_solution(scaled_model, m) display_results(m) add_costing(m) display_costing(m) return m, results
[docs] def build(): """ Build and connect the unit model blocks present in the University of Kentucky REE processing plant. """ m = ConcreteModel() m.fs = FlowsheetBlock(dynamic=False) # Leaching property and unit models m.fs.leach_soln = LeachSolutionParameters() m.fs.coal = CoalRefuseParameters() m.fs.leach_rxns = CoalRefuseLeachingReactions() m.fs.leach = LeachingTrain( number_of_tanks=2, liquid_phase={ "property_package": m.fs.leach_soln, "has_energy_balance": False, "has_pressure_balance": False, }, solid_phase={ "property_package": m.fs.coal, "has_energy_balance": False, "has_pressure_balance": False, }, reaction_package=m.fs.leach_rxns, ) m.fs.sl_sep1 = SLSeparator( solid_property_package=m.fs.coal, liquid_property_package=m.fs.leach_soln, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.leach_mixer = Mixer( property_package=m.fs.leach_soln, num_inlets=3, inlet_list=["load_recycle", "scrub_recycle", "feed"], material_balance_type=MaterialBalanceType.componentTotal, energy_mixing_type=MixingType.none, momentum_mixing_type=MomentumMixingType.none, ) m.fs.leach_liquid_feed = Feed(property_package=m.fs.leach_soln) m.fs.leach_solid_feed = Feed(property_package=m.fs.coal) m.fs.leach_filter_cake = Product(property_package=m.fs.coal) m.fs.leach_filter_cake_liquid = Product(property_package=m.fs.leach_soln) # ---------------------------------------------------------------------------------------------------------------- # Solvent extraction property and unit models m.fs.prop_o = REESolExOgParameters() m.fs.rougher_org_make_up = Feed(property_package=m.fs.prop_o) m.fs.solex_rougher_load = SolventExtraction( number_of_finite_elements=3, dynamic=False, aqueous_stream={ "property_package": m.fs.leach_soln, "flow_direction": FlowDirection.forward, "has_energy_balance": False, "has_pressure_balance": False, }, organic_stream={ "property_package": m.fs.prop_o, "flow_direction": FlowDirection.backward, "has_energy_balance": False, "has_pressure_balance": False, }, aqueous_to_organic=True, ) m.fs.acid_feed1 = Feed(property_package=m.fs.leach_soln) m.fs.solex_rougher_scrub = SolventExtraction( number_of_finite_elements=1, dynamic=False, aqueous_stream={ "property_package": m.fs.leach_soln, "flow_direction": FlowDirection.backward, "has_energy_balance": False, "has_pressure_balance": False, }, organic_stream={ "property_package": m.fs.prop_o, "flow_direction": FlowDirection.forward, "has_energy_balance": False, "has_pressure_balance": False, }, aqueous_to_organic=False, ) m.fs.acid_feed2 = Feed(property_package=m.fs.leach_soln) m.fs.solex_rougher_strip = SolventExtraction( number_of_finite_elements=2, dynamic=False, aqueous_stream={ "property_package": m.fs.leach_soln, "flow_direction": FlowDirection.backward, "has_energy_balance": False, "has_pressure_balance": False, }, organic_stream={ "property_package": m.fs.prop_o, "flow_direction": FlowDirection.forward, "has_energy_balance": False, "has_pressure_balance": False, }, aqueous_to_organic=False, ) m.fs.rougher_sep = Separator( property_package=m.fs.prop_o, outlet_list=["recycle", "purge"], split_basis=SplittingType.totalFlow, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.rougher_mixer = Mixer( property_package=m.fs.prop_o, num_inlets=2, inlet_list=["make_up", "recycle"], material_balance_type=MaterialBalanceType.componentTotal, energy_mixing_type=MixingType.none, momentum_mixing_type=MomentumMixingType.none, ) m.fs.load_sep = Separator( property_package=m.fs.leach_soln, outlet_list=["recycle", "purge"], split_basis=SplittingType.totalFlow, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.scrub_sep = Separator( property_package=m.fs.leach_soln, outlet_list=["recycle", "purge"], split_basis=SplittingType.totalFlow, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.sc_circuit_purge = Product(property_package=m.fs.prop_o) m.fs.solex_cleaner_load = SolventExtraction( number_of_finite_elements=3, dynamic=False, aqueous_stream={ "property_package": m.fs.leach_soln, "flow_direction": FlowDirection.forward, "has_energy_balance": False, "has_pressure_balance": False, }, organic_stream={ "property_package": m.fs.prop_o, "flow_direction": FlowDirection.backward, "has_energy_balance": False, "has_pressure_balance": False, }, aqueous_to_organic=True, ) m.fs.solex_cleaner_strip = SolventExtraction( number_of_finite_elements=3, dynamic=False, aqueous_stream={ "property_package": m.fs.leach_soln, "flow_direction": FlowDirection.backward, "has_energy_balance": False, "has_pressure_balance": False, }, organic_stream={ "property_package": m.fs.prop_o, "flow_direction": FlowDirection.forward, "has_energy_balance": False, "has_pressure_balance": False, }, aqueous_to_organic=False, ) m.fs.cleaner_org_make_up = Feed(property_package=m.fs.prop_o) m.fs.cleaner_mixer = Mixer( property_package=m.fs.prop_o, num_inlets=2, inlet_list=["make_up", "recycle"], material_balance_type=MaterialBalanceType.componentTotal, energy_mixing_type=MixingType.none, momentum_mixing_type=MomentumMixingType.none, ) m.fs.cleaner_sep = Separator( property_package=m.fs.prop_o, outlet_list=["recycle", "purge"], split_basis=SplittingType.totalFlow, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.leach_sx_mixer = Mixer( property_package=m.fs.leach_soln, num_inlets=2, inlet_list=["leach", "cleaner"], material_balance_type=MaterialBalanceType.componentTotal, energy_mixing_type=MixingType.none, momentum_mixing_type=MomentumMixingType.none, ) m.fs.acid_feed3 = Feed(property_package=m.fs.leach_soln) m.fs.cleaner_purge = Product(property_package=m.fs.prop_o) # -------------------------------------------------------------------------------------------------------------- # Precipitation property and unit models m.fs.properties_aq = AqueousParameter() m.fs.properties_solid = PrecipitateParameters() m.fs.precipitator = Precipitator( property_package_aqueous=m.fs.properties_aq, property_package_precipitate=m.fs.properties_solid, ) m.fs.sl_sep2 = SLSeparator( solid_property_package=m.fs.properties_solid, liquid_property_package=m.fs.leach_soln, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.precip_sep = Separator( property_package=m.fs.leach_soln, outlet_list=["recycle", "purge"], split_basis=SplittingType.totalFlow, material_balance_type=MaterialBalanceType.componentTotal, momentum_balance_type=MomentumBalanceType.none, energy_split_basis=EnergySplittingType.none, ) m.fs.precip_sx_mixer = Mixer( property_package=m.fs.leach_soln, num_inlets=2, inlet_list=["precip", "rougher"], material_balance_type=MaterialBalanceType.componentTotal, energy_mixing_type=MixingType.none, momentum_mixing_type=MomentumMixingType.none, ) m.fs.precip_purge = Product(property_package=m.fs.properties_aq) # ----------------------------------------------------------------------------------------------------------------- # Roasting property and unit models gas_species = {"O2", "H2O", "CO2", "N2"} m.fs.prop_gas = GenericParameterBlock( **get_prop(gas_species, ["Vap"], EosType.IDEAL), doc="gas property", ) m.fs.prop_solid = PrecipitateParameters() m.fs.roaster = REEOxalateRoaster( property_package_gas=m.fs.prop_gas, property_package_precipitate_solid=m.fs.prop_solid, property_package_precipitate_liquid=m.fs.properties_aq, has_holdup=False, has_heat_transfer=True, has_pressure_change=True, ) # ----------------------------------------------------------------------------------------------------------------- # UKy flowsheet connections m.fs.leaching_sol_feed = Arc( source=m.fs.leach_solid_feed.outlet, destination=m.fs.leach.solid_inlet ) m.fs.leaching_liq_feed = Arc( source=m.fs.leach_liquid_feed.outlet, destination=m.fs.leach_mixer.feed ) m.fs.leaching_feed_mixture = Arc( source=m.fs.leach_mixer.outlet, destination=m.fs.leach.liquid_inlet ) m.fs.leaching_solid_outlet = Arc( source=m.fs.leach.solid_outlet, destination=m.fs.sl_sep1.solid_inlet ) m.fs.leaching_liquid_outlet = Arc( source=m.fs.leach.liquid_outlet, destination=m.fs.sl_sep1.liquid_inlet ) m.fs.sl_sep1_solid_outlet = Arc( source=m.fs.sl_sep1.solid_outlet, destination=m.fs.leach_filter_cake.inlet ) m.fs.sl_sep1_retained_liquid_outlet = Arc( source=m.fs.sl_sep1.retained_liquid_outlet, destination=m.fs.leach_filter_cake_liquid.inlet, ) m.fs.sl_sep1_liquid_outlet = Arc( source=m.fs.sl_sep1.recovered_liquid_outlet, destination=m.fs.leach_sx_mixer.leach, ) m.fs.sx_rougher_load_aq_feed = Arc( source=m.fs.leach_sx_mixer.outlet, destination=m.fs.solex_rougher_load.aqueous_inlet, ) m.fs.sx_rougher_org_feed = Arc( source=m.fs.rougher_org_make_up.outlet, destination=m.fs.rougher_mixer.make_up ) m.fs.sx_rougher_mixed_org_recycle = Arc( source=m.fs.rougher_mixer.outlet, destination=m.fs.solex_rougher_load.organic_inlet, ) m.fs.sx_rougher_load_aq_outlet = Arc( source=m.fs.solex_rougher_load.aqueous_outlet, destination=m.fs.load_sep.inlet, ) m.fs.sx_rougher_load_aq_recycle = Arc( source=m.fs.load_sep.recycle, destination=m.fs.leach_mixer.load_recycle ) m.fs.sx_rougher_load_org_outlet = Arc( source=m.fs.solex_rougher_load.organic_outlet, destination=m.fs.solex_rougher_scrub.organic_inlet, ) m.fs.sx_rougher_scrub_acid_feed = Arc( source=m.fs.acid_feed1.outlet, destination=m.fs.solex_rougher_scrub.aqueous_inlet, ) m.fs.sx_rougher_scrub_aq_outlet = Arc( source=m.fs.solex_rougher_scrub.aqueous_outlet, destination=m.fs.scrub_sep.inlet, ) m.fs.sx_rougher_scrub_aq_recycle = Arc( source=m.fs.scrub_sep.recycle, destination=m.fs.leach_mixer.scrub_recycle ) m.fs.sx_rougher_scrub_org_outlet = Arc( source=m.fs.solex_rougher_scrub.organic_outlet, destination=m.fs.solex_rougher_strip.organic_inlet, ) m.fs.sx_rougher_strip_acid_feed = Arc( source=m.fs.acid_feed2.outlet, destination=m.fs.solex_rougher_strip.aqueous_inlet, ) m.fs.sx_rougher_strip_org_outlet = Arc( source=m.fs.solex_rougher_strip.organic_outlet, destination=m.fs.rougher_sep.inlet, ) m.fs.sx_rougher_strip_org_purge = Arc( source=m.fs.rougher_sep.purge, destination=m.fs.sc_circuit_purge.inlet ) m.fs.sx_rougher_strip_org_recycle = Arc( source=m.fs.rougher_sep.recycle, destination=m.fs.rougher_mixer.recycle ) m.fs.sx_rougher_strip_aq_outlet = Arc( source=m.fs.solex_rougher_strip.aqueous_outlet, destination=m.fs.precip_sx_mixer.rougher, ) m.fs.sx_cleaner_load_aq_feed = Arc( source=m.fs.precip_sx_mixer.outlet, destination=m.fs.solex_cleaner_load.aqueous_inlet, ) m.fs.sx_cleaner_org_feed = Arc( source=m.fs.cleaner_org_make_up.outlet, destination=m.fs.cleaner_mixer.make_up ) m.fs.sx_cleaner_mixed_org_recycle = Arc( source=m.fs.cleaner_mixer.outlet, destination=m.fs.solex_cleaner_load.organic_inlet, ) m.fs.sx_cleaner_load_aq_outlet = Arc( source=m.fs.solex_cleaner_load.aqueous_outlet, destination=m.fs.leach_sx_mixer.cleaner, ) m.fs.sx_cleaner_strip_acid_feed = Arc( source=m.fs.acid_feed3.outlet, destination=m.fs.solex_cleaner_strip.aqueous_inlet, ) m.fs.sx_cleaner_load_org_outlet = Arc( source=m.fs.solex_cleaner_load.organic_outlet, destination=m.fs.solex_cleaner_strip.organic_inlet, ) m.fs.sx_cleaner_strip_org_outlet = Arc( source=m.fs.solex_cleaner_strip.organic_outlet, destination=m.fs.cleaner_sep.inlet, ) m.fs.sx_cleaner_strip_org_purge = Arc( source=m.fs.cleaner_sep.purge, destination=m.fs.cleaner_purge.inlet ) m.fs.sx_cleaner_strip_org_recycle = Arc( source=m.fs.cleaner_sep.recycle, destination=m.fs.cleaner_mixer.recycle ) m.fs.sx_cleaner_strip_aq_outlet = Arc( source=m.fs.solex_cleaner_strip.aqueous_outlet, destination=m.fs.precipitator.aqueous_inlet, ) m.fs.precip_solid_outlet = Arc( source=m.fs.precipitator.precipitate_outlet, destination=m.fs.sl_sep2.solid_inlet, ) m.fs.precip_aq_outlet = Arc( source=m.fs.precipitator.aqueous_outlet, destination=m.fs.sl_sep2.liquid_inlet ) m.fs.sl_sep2_solid_outlet = Arc( source=m.fs.sl_sep2.solid_outlet, destination=m.fs.roaster.solid_inlet ) m.fs.sl_sep2_retained_liquid_outlet = Arc( source=m.fs.sl_sep2.retained_liquid_outlet, destination=m.fs.roaster.liquid_inlet, ) m.fs.sl_sep2_liquid_outlet = Arc( source=m.fs.sl_sep2.recovered_liquid_outlet, destination=m.fs.precip_sep.inlet ) m.fs.sl_sep2_aq_purge = Arc( source=m.fs.precip_sep.purge, destination=m.fs.precip_purge.inlet ) m.fs.sl_sep2_aq_recycle = Arc( source=m.fs.precip_sep.recycle, destination=m.fs.precip_sx_mixer.precip, ) TransformationFactory("network.expand_arcs").apply_to(m) return m
[docs] def set_partition_coefficients(m): """ Sets the partition coefficients for each finite element in the solvent extraction blocks. Args: m: pyomo model """ m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Al"] = ( 5.2 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Ca"] = ( 3.0 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Fe"] = ( 24.7 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Sc"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Y"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "La"] = ( 32.4 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Ce"] = ( 58.2 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Pr"] = ( 58.2 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Nd"] = ( 87.6 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Sm"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Gd"] = ( 69.8 / 100 ) m.fs.solex_rougher_load.partition_coefficient[1, "aqueous", "organic", "Dy"] = ( 96.6 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Al"] = ( 4.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Ca"] = ( 12.3 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Fe"] = ( 6.4 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Sc"] = ( 16.7 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Y"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "La"] = ( 23.2 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Ce"] = ( 24.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Pr"] = ( 15.1 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Nd"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Sm"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Gd"] = ( 7.6 / 100 ) m.fs.solex_rougher_load.partition_coefficient[2, "aqueous", "organic", "Dy"] = ( 5.0 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Al"] = ( 4.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Ca"] = ( 12.3 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Fe"] = ( 6.4 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Sc"] = ( 16.7 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Y"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "La"] = ( 23.2 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Ce"] = ( 24.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Pr"] = ( 15.1 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Nd"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Sm"] = ( 99.9 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Gd"] = ( 7.6 / 100 ) m.fs.solex_rougher_load.partition_coefficient[3, "aqueous", "organic", "Dy"] = ( 5.0 / 100 ) m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Al"] = ( 100 - 0.12 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Ca"] = ( 100 - 0.55 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Fe"] = ( 100 - 0.007 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Sc"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Y"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "La"] = ( 100 - 99.8 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Ce"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Pr"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Nd"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Sm"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Gd"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_scrub.partition_coefficient[1, "aqueous", "organic", "Dy"] = ( 100 - 99.9 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Al"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Ca"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Fe"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Sc"] = ( 100 - 98.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Y"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "La"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Ce"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Pr"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Nd"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Sm"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Gd"] = ( 100 - 0.5 ) / 100 m.fs.solex_rougher_strip.partition_coefficient[:, "aqueous", "organic", "Dy"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Al"] = ( 3.6 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Ca"] = ( 3.7 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Fe"] = ( 2.1 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Sc"] = ( 99.9 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Y"] = ( 99.9 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "La"] = ( 75.2 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Ce"] = ( 95.7 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Pr"] = ( 96.5 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Nd"] = ( 99.2 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Sm"] = ( 99.9 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Gd"] = ( 98.6 / 100 ) m.fs.solex_cleaner_load.partition_coefficient[:, "aqueous", "organic", "Dy"] = ( 99.9 / 100 ) m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Al"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Ca"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Fe"] = ( 100 - 5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Sc"] = ( 100 - 98.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Y"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "La"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Ce"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Pr"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Nd"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Sm"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Gd"] = ( 100 - 0.5 ) / 100 m.fs.solex_cleaner_strip.partition_coefficient[:, "aqueous", "organic", "Dy"] = ( 100 - 0.5 ) / 100
[docs] def set_scaling(m): """ Set the scaling factors to improve solver performance. Args: m: pyomo model """ # Scaling m.scaling_factor = Suffix(direction=Suffix.EXPORT) aqueous_component_set = [ "H2O", "H", "HSO4", "SO4", "Cl", "Sc", "Y", "La", "Ce", "Pr", "Nd", "Sm", "Gd", "Dy", "Al", "Ca", "Fe", ] organic_component_set = [ "Sc", "Y", "La", "Ce", "Pr", "Nd", "Sm", "Gd", "Dy", "Al", "Ca", "Fe", ] for component in aqueous_component_set: m.scaling_factor[ m.fs.leach.mscontactor.liquid[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_liquid_feed.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.sl_sep1.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.sl_sep1.split.recovered_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.sl_sep1.split.retained_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_filter_cake_liquid.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_mixer.load_recycle_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_mixer.scrub_recycle_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.leach_mixer.feed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.leach_mixer.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_scrub.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.aqueous[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.aqueous[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.aqueous_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[m.fs.acid_feed1.properties[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.solex_rougher_scrub.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_scrub.mscontactor.aqueous_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[m.fs.acid_feed2.properties[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.aqueous[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.aqueous_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.aqueous[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.aqueous[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.aqueous_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.leach_sx_mixer.leach_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_sx_mixer.cleaner_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach_sx_mixer.mixed_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.aqueous_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.sl_sep2.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.sl_sep2.split.retained_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.sl_sep2.split.recovered_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.load_sep.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.load_sep.recycle_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.load_sep.purge_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.scrub_sep.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.scrub_sep.recycle_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.scrub_sep.purge_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.precip_sep.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.precip_sep.recycle_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[m.fs.precip_sep.purge_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.precip_purge.properties[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.precip_sx_mixer.precip_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.precip_sx_mixer.rougher_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.precip_sx_mixer.mixed_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.acid_feed3.properties[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.precip_purge.properties[0].conc_mol_comp[component]] = 1 m.scaling_factor[ m.fs.precipitator.cv_aqueous.properties_in[0].conc_mol_comp[component] ] = 1 m.scaling_factor[ m.fs.precipitator.cv_aqueous.properties_out[0].conc_mol_comp[component] ] = 1 for component in organic_component_set: m.scaling_factor[ m.fs.rougher_org_make_up.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.organic[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.organic[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_load.mscontactor.organic_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_scrub.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_scrub.mscontactor.organic_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.organic[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_rougher_strip.mscontactor.organic_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.rougher_mixer.make_up_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.rougher_mixer.recycle_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.rougher_mixer.mixed_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[m.fs.rougher_sep.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.rougher_sep.recycle_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[m.fs.rougher_sep.purge_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.rougher_mixer.make_up_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.rougher_mixer.recycle_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.rougher_mixer.mixed_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[ m.fs.sc_circuit_purge.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.cleaner_mixer.make_up_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.cleaner_mixer.recycle_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.cleaner_mixer.mixed_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[ m.fs.sc_circuit_purge.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.organic[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.organic[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.organic_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_load.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.cleaner_sep.mixed_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[m.fs.cleaner_sep.recycle_state[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[m.fs.cleaner_sep.purge_state[0].conc_mol_comp[component]] = 1e5 m.scaling_factor[ m.fs.cleaner_org_make_up.properties[0].conc_mol_comp[component] ] = 1e5 m.scaling_factor[m.fs.cleaner_purge.properties[0].conc_mol_comp[component]] = ( 1e5 ) m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.organic[0, 1].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.organic[0, 2].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.organic[0, 3].conc_mol_comp[component] ] = 1e5 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.organic_inlet_state[0].conc_mol_comp[ component ] ] = 1e5 m.scaling_factor[m.fs.solex_cleaner_load.mscontactor.aqueous[0, 1].flow_vol] = 1e-2 m.scaling_factor[m.fs.solex_cleaner_load.mscontactor.organic[0, 1].flow_vol] = 1e-2 m.scaling_factor[m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 1].flow_vol] = 1e-2 m.scaling_factor[m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 2].flow_vol] = 1e-2 m.scaling_factor[m.fs.solex_cleaner_strip.mscontactor.aqueous[0, 3].flow_vol] = 1e-2 m.scaling_factor[ m.fs.solex_cleaner_strip.mscontactor.aqueous_inlet_state[0].flow_vol ] = 1e-2 m.scaling_factor[m.fs.solex_cleaner_strip.mscontactor.organic[0, 1].flow_vol] = 1e-2 m.scaling_factor[m.fs.sl_sep2.solid_state[0].temperature] = 1e-2 m.scaling_factor[m.fs.sl_sep2.liquid_inlet_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.sl_sep2.split.recovered_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.sl_sep2.split.retained_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precip_sep.mixed_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precip_sep.recycle_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precip_sep.purge_state[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precip_purge.properties[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precipitator.cv_precipitate[0].temperature] = 1e2 m.scaling_factor[m.fs.precipitator.cv_aqueous.properties_in[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.precipitator.cv_aqueous.properties_out[0].flow_vol] = 1e-2 m.scaling_factor[m.fs.roaster.gas_in[0].flow_mol] = 1e-3 m.scaling_factor[m.fs.roaster.gas_in[0].flow_mol_phase["Vap"]] = 1e-3 m.scaling_factor[m.fs.roaster.gas_in[0].temperature] = 1e-2 m.scaling_factor[m.fs.roaster.gas_in[0].pressure] = 1e-5 m.scaling_factor[m.fs.roaster.gas_out[0].flow_mol_phase["Vap"]] = 1e-3 m.scaling_factor[m.fs.roaster.gas_out[0].flow_mol] = 1e-3 m.scaling_factor[m.fs.roaster.gas_out[0].temperature] = 1e-2 m.scaling_factor[m.fs.roaster.gas_out[0].pressure] = 1e-5 m.scaling_factor[m.fs.roaster.solid_in[0].temperature] = 1e-2 return m
[docs] def set_operating_conditions(m): """ Set the operating conditions of the flowsheet such that the degrees of freedom are zero. Args: m: pyomo model """ m.fs.leach_liquid_feed.flow_vol.fix(224.3 * units.L / units.hour) m.fs.leach_liquid_feed.conc_mass_comp.fix(1e-10 * units.mg / units.L) m.fs.leach_liquid_feed.conc_mass_comp[0, "H"].fix( 2 * 0.05 * 1e3 * units.mg / units.L ) m.fs.leach_liquid_feed.conc_mass_comp[0, "HSO4"].fix(1e-8 * units.mg / units.L) m.fs.leach_liquid_feed.conc_mass_comp[0, "SO4"].fix( 0.05 * 96e3 * units.mg / units.L ) m.fs.leach_solid_feed.flow_mass.fix(22.68 * units.kg / units.hour) m.fs.leach_solid_feed.mass_frac_comp[0, "inerts"].fix(0.6952 * units.kg / units.kg) m.fs.leach_solid_feed.mass_frac_comp[0, "Al2O3"].fix(0.237 * units.kg / units.kg) m.fs.leach_solid_feed.mass_frac_comp[0, "Fe2O3"].fix(0.0642 * units.kg / units.kg) m.fs.leach_solid_feed.mass_frac_comp[0, "CaO"].fix(3.31e-3 * units.kg / units.kg) m.fs.leach_solid_feed.mass_frac_comp[0, "Sc2O3"].fix( 2.77966e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Y2O3"].fix( 3.28653e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "La2O3"].fix( 6.77769e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Ce2O3"].fix( 0.000156161 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Pr2O3"].fix( 1.71438e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Nd2O3"].fix( 6.76618e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Sm2O3"].fix( 1.47926e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Gd2O3"].fix( 1.0405e-05 * units.kg / units.kg ) m.fs.leach_solid_feed.mass_frac_comp[0, "Dy2O3"].fix( 7.54827e-06 * units.kg / units.kg ) m.fs.leach.volume.fix(100 * units.gallon) m.fs.load_sep.split_fraction[:, "recycle"].fix(0.9) m.fs.scrub_sep.split_fraction[:, "recycle"].fix(0.9) m.fs.rougher_org_make_up.flow_vol.fix(6.201) m.fs.rougher_org_make_up.conc_mass_comp[0, "Al"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Ca"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Fe"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Sc"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Y"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "La"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Ce"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Pr"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Nd"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Sm"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Gd"].fix(eps) m.fs.rougher_org_make_up.conc_mass_comp[0, "Dy"].fix(eps) m.fs.acid_feed1.flow_vol.fix(0.09) m.fs.acid_feed1.conc_mass_comp[0, "H2O"].fix(1000000) m.fs.acid_feed1.conc_mass_comp[0, "H"].fix(10.36) m.fs.acid_feed1.conc_mass_comp[0, "SO4"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "HSO4"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Cl"].fix(359.64) m.fs.acid_feed1.conc_mass_comp[0, "Al"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Ca"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Fe"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Sc"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Y"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "La"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Ce"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Pr"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Nd"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Sm"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Gd"].fix(eps) m.fs.acid_feed1.conc_mass_comp[0, "Dy"].fix(eps) m.fs.acid_feed2.flow_vol.fix(0.09) m.fs.acid_feed2.conc_mass_comp[0, "H2O"].fix(1000000) m.fs.acid_feed2.conc_mass_comp[0, "H"].fix( 10.36 * 4 ) # Arbitrarily choose 4x the dilute solution m.fs.acid_feed2.conc_mass_comp[0, "SO4"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "HSO4"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Cl"].fix(359.64 * 4) m.fs.acid_feed2.conc_mass_comp[0, "Al"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Ca"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Fe"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Sc"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Y"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "La"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Ce"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Pr"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Nd"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Sm"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Gd"].fix(eps) m.fs.acid_feed2.conc_mass_comp[0, "Dy"].fix(eps) m.fs.rougher_sep.split_fraction[:, "recycle"].fix(0.9) m.fs.acid_feed3.flow_vol.fix(9) m.fs.acid_feed3.conc_mass_comp[0, "H2O"].fix(1000000) m.fs.acid_feed3.conc_mass_comp[0, "H"].fix( 10.36 * 4 ) # Arbitrarily choose 4x the dilute solution m.fs.acid_feed3.conc_mass_comp[0, "SO4"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "HSO4"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Cl"].fix(359.64 * 4) m.fs.acid_feed3.conc_mass_comp[0, "Al"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Ca"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Fe"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Sc"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Y"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "La"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Ce"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Pr"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Nd"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Sm"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Gd"].fix(eps) m.fs.acid_feed3.conc_mass_comp[0, "Dy"].fix(eps) m.fs.cleaner_org_make_up.flow_vol.fix(6.201) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Al"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Ca"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Fe"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Sc"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Y"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "La"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Ce"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Pr"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Nd"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Sm"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Gd"].fix(eps) m.fs.cleaner_org_make_up.conc_mass_comp[0, "Dy"].fix(eps) m.fs.cleaner_sep.split_fraction[:, "recycle"].fix(0.9) m.fs.sl_sep1.liquid_recovery.fix(0.7) # TODO: Set sl_sep2 recovery to 0.95 and resolve resultant initialization issues m.fs.sl_sep2.liquid_recovery.fix(0.88) m.fs.precipitator.cv_precipitate[0].temperature.fix(348.15 * units.K) m.fs.precip_sep.split_fraction[:, "recycle"].fix(0.9) # Roaster gas feed m.fs.roaster.deltaP.fix(0) m.fs.roaster.gas_inlet.temperature.fix(1330) m.fs.roaster.gas_inlet.pressure.fix(101325) # Inlet flue gas mole flow rate fgas = 0.00781 # Inlet flue gas composition, typical flue gas by burning CH4 with air with stoichiometric ratio of 2.3 gas_comp = { "O2": 0.1118, "H2O": 0.1005, "CO2": 0.0431, "N2": 0.7446, } for i, v in gas_comp.items(): m.fs.roaster.gas_inlet.mole_frac_comp[0, i].fix(v) m.fs.roaster.gas_inlet.flow_mol.fix(fgas) # Fix outlet product temperature m.fs.roaster.gas_outlet.temperature.fix(873.15) # Fix operating conditions m.fs.roaster.frac_comp_recovery.fix(0.95) # Touch properties that are used in the UI m.fs.leach.mscontactor.solid_inlet_state[0].flow_mass m.fs.leach.mscontactor.solid_inlet_state[0].mass_frac_comp m.fs.leach.mscontactor.liquid_inlet_state[0].flow_vol m.fs.leach.mscontactor.liquid_inlet_state[0].conc_mol_comp m.fs.solex_cleaner_load.mscontactor.organic_inlet_state[0].conc_mass_comp m.fs.solex_cleaner_strip.mscontactor.organic_inlet_state[0].conc_mass_comp m.fs.solex_rougher_load.mscontactor.organic_inlet_state[0].conc_mass_comp m.fs.solex_rougher_strip.mscontactor.organic_inlet_state[0].conc_mass_comp m.fs.solex_rougher_scrub.mscontactor.organic_inlet_state[0].conc_mass_comp m.fs.solex_cleaner_load.mscontactor.aqueous_inlet_state[0].conc_mass_comp m.fs.solex_cleaner_strip.mscontactor.aqueous_inlet_state[0].conc_mass_comp m.fs.solex_rougher_load.mscontactor.aqueous_inlet_state[0].conc_mass_comp m.fs.solex_rougher_strip.mscontactor.aqueous_inlet_state[0].conc_mass_comp m.fs.solex_rougher_scrub.mscontactor.aqueous_inlet_state[0].conc_mass_comp m.fs.precipitator.cv_aqueous.properties_out[0].flow_vol m.fs.precipitator.cv_aqueous.properties_out[0].conc_mass_comp m.fs.precipitator.cv_precipitate[0].temperature m.fs.precipitator.cv_precipitate[0].flow_mol_comp
[docs] def initialize_system(m): """ Provide initialized values for all streams in the system. Args: m: pyomo model """ seq = SequentialDecomposition() seq.options.tear_method = "Direct" seq.options.iterLim = 1 seq.options.tear_set = [ m.fs.leaching_feed_mixture, m.fs.sx_rougher_load_aq_feed, m.fs.sx_rougher_mixed_org_recycle, m.fs.sx_cleaner_load_aq_feed, m.fs.sx_cleaner_mixed_org_recycle, ] G = seq.create_graph(m) order = seq.calculation_order(G) print("Initialization Order") for o in order: print(o[0].name) tear_guesses1 = { "flow_vol": {0: 747.99}, "conc_mass_comp": { (0, "Al"): 180.84, (0, "Ca"): 28.93, (0, "Ce"): 5.48, (0, "Dy"): 4.46e-11, (0, "Fe"): 269.98, (0, "Gd"): 2.60e-7, (0, "H"): 20.06, (0, "H2O"): 1000000, (0, "HSO4"): 963.06, (0, "Cl"): 1e-8, (0, "La"): 0.0037, (0, "Nd"): 1.81e-7, (0, "Pr"): 3.65e-6, (0, "SO4"): 486.24, (0, "Sc"): 4.17e-11, (0, "Sm"): 6.30e-10, (0, "Y"): 7.18e-11, }, } tear_guesses2 = { "flow_vol": {0: 62.01}, "conc_mass_comp": { (0, "Al"): 1e-9, (0, "Ca"): 1e-9, (0, "Ce"): 1e-4, (0, "Dy"): 1e-7, (0, "Fe"): 1e-7, (0, "Gd"): 1e-6, (0, "La"): 1e-5, (0, "Nd"): 1e-4, (0, "Pr"): 1e-6, (0, "Sc"): 250, (0, "Sm"): 1e-6, (0, "Y"): 1e-6, }, } tear_guesses3 = { "flow_vol": {0: 520}, "conc_mass_comp": { (0, "Al"): 430, (0, "Ca"): 99, (0, "Ce"): 2, (0, "Dy"): 0.01, (0, "Fe"): 660, (0, "Gd"): 0.1, (0, "H"): 2, (0, "H2O"): 1000000, (0, "HSO4"): 900, (0, "Cl"): 0.1, (0, "La"): 1, (0, "Nd"): 1, (0, "Pr"): 0.1, (0, "SO4"): 4000, (0, "Sc"): 0.05, (0, "Sm"): 0.07, (0, "Y"): 0.1, }, } tear_guesses4 = { "flow_vol": {0: 64}, "conc_mass_comp": { (0, "Al"): 1e-9, (0, "Ca"): 1e-9, (0, "Ce"): 1e-5, (0, "Dy"): 1e-7, (0, "Fe"): 1e-7, (0, "Gd"): 1e-6, (0, "La"): 1e-5, (0, "Nd"): 1e-5, (0, "Pr"): 1e-6, (0, "Sc"): 321.34, (0, "Sm"): 1e-6, (0, "Y"): 1e-6, }, } tear_guesses5 = { "flow_vol": {0: 5.7}, "conc_mass_comp": { (0, "Al"): 5, (0, "Ca"): 16, (0, "Ce"): 346, (0, "Dy"): 6, (0, "Fe"): 1, (0, "Gd"): 22, (0, "H"): 14, (0, "H2O"): 1000000, (0, "HSO4"): 1e-7, (0, "Cl"): 1400, (0, "La"): 160, (0, "Nd"): 121, (0, "Pr"): 30, (0, "SO4"): 1e-7, (0, "Sc"): 149.2, (0, "Sm"): 13, (0, "Y"): 18, }, } # Pass the tear_guess to the SD tool seq.set_guesses_for(m.fs.leach.liquid_inlet, tear_guesses1) seq.set_guesses_for(m.fs.solex_rougher_load.organic_inlet, tear_guesses2) seq.set_guesses_for(m.fs.solex_rougher_load.aqueous_inlet, tear_guesses3) seq.set_guesses_for(m.fs.solex_cleaner_load.organic_inlet, tear_guesses4) seq.set_guesses_for(m.fs.solex_cleaner_load.aqueous_inlet, tear_guesses5) initializer_feed = FeedInitializer() feed_units = [ m.fs.leach_liquid_feed, m.fs.leach_solid_feed, m.fs.rougher_org_make_up, m.fs.acid_feed1, m.fs.acid_feed2, m.fs.acid_feed3, m.fs.cleaner_org_make_up, ] initializer_product = ProductInitializer() product_units = [ m.fs.leach_filter_cake, m.fs.leach_filter_cake_liquid, m.fs.cleaner_purge, m.fs.sc_circuit_purge, m.fs.precip_purge, ] initializer_sep = SeparatorInitializer() sep_units = [ m.fs.scrub_sep, m.fs.precip_sep, m.fs.cleaner_sep, m.fs.rougher_sep, ] initializer_mix = MixerInitializer() mix_units = [ m.fs.precip_sx_mixer, m.fs.cleaner_mixer, m.fs.rougher_mixer, ] initializer_sx = SolventExtractionInitializer() sx_units = [ m.fs.solex_rougher_load, m.fs.solex_rougher_scrub, m.fs.solex_cleaner_load, m.fs.solex_cleaner_strip, ] initializer_bt = BlockTriangularizationInitializer() def function(unit): if unit in feed_units: _log.info(f"Initializing {unit}") initializer_feed.initialize(unit) elif unit in product_units: _log.info(f"Initializing {unit}") initializer_product.initialize(unit) elif unit in sep_units: _log.info(f"Initializing {unit}") initializer_sep.initialize(unit) elif unit in mix_units: _log.info(f"Initializing {unit}") initializer_mix.initialize(unit) elif unit in sx_units: _log.info(f"Initializing {unit}") initializer_sx.initialize(unit) elif unit == m.fs.leach: _log.info(f"Initializing {unit}") # Fix feed states m.fs.leach.liquid_inlet.flow_vol.fix() m.fs.leach.liquid_inlet.conc_mass_comp.fix() m.fs.leach.solid_inlet.flow_mass.fix() m.fs.leach.solid_inlet.mass_frac_comp.fix() # Re-solve unit solver = SolverFactory("ipopt") solver.solve(m.fs.leach, tee=True) # Unfix feed states m.fs.leach.liquid_inlet.flow_vol.unfix() m.fs.leach.liquid_inlet.conc_mass_comp.unfix() m.fs.leach.solid_inlet.flow_mass.unfix() m.fs.leach.solid_inlet.mass_frac_comp.unfix() else: _log.info(f"Initializing {unit}") initializer_bt.initialize(unit) seq.run(m, function)
[docs] def solve_system(m, solver=None, tee=False): """ Solve the model. Args: m: pyomo model solver: optimization solver tee: boolean indicator to stream IPOPT solution """ if hasattr(solver, "solve"): solver = solver else: solver = get_solver() results = solver.solve(m, tee=tee) return results
[docs] def fix_organic_recycle(m): """ Fix the volumetric flow rate of the organic recycle streams and unfix the flow of make-up streams. Args: m: pyomo model """ m.fs.rougher_org_make_up.outlet.flow_vol.unfix() m.fs.rougher_mixer.outlet.flow_vol.fix(62.01) m.fs.cleaner_org_make_up.outlet.flow_vol.unfix() m.fs.cleaner_mixer.outlet.flow_vol.fix(62.01)
[docs] def display_results(m): """ Print key flowsheet outputs. Args: m: pyomo model """ m.fs.roaster.report() metal_mass_frac = { "Al2O3": 26.98 * 2 / (26.98 * 2 + 16 * 3), "Fe2O3": 55.845 * 2 / (55.845 * 2 + 16 * 3), "CaO": 40.078 / (40.078 + 16), "Sc2O3": 44.956 * 2 / (44.956 * 2 + 16 * 3), "Y2O3": 88.906 * 2 / (88.906 * 2 + 16 * 3), "La2O3": 138.91 * 2 / (138.91 * 2 + 16 * 3), "Ce2O3": 140.12 * 2 / (140.12 * 2 + 16 * 3), "Pr2O3": 140.91 * 2 / (140.91 * 2 + 16 * 3), "Nd2O3": 144.24 * 2 / (144.24 * 2 + 16 * 3), "Sm2O3": 150.36 * 2 / (150.36 * 2 + 16 * 3), "Gd2O3": 157.25 * 2 / (157.25 * 2 + 16 * 3), "Dy2O3": 162.5 * 2 / (162.5 * 2 + 16 * 3), } molar_mass = { "Al2O3": (26.98 * 2 + 16 * 3) * units.g / units.mol, "Fe2O3": (55.845 * 2 + 16 * 3) * units.g / units.mol, "CaO": (40.078 + 16) * units.g / units.mol, "Sc2O3": (44.956 * 2 + 16 * 3) * units.g / units.mol, "Y2O3": (88.906 * 2 + 16 * 3) * units.g / units.mol, "La2O3": (138.91 * 2 + 16 * 3) * units.g / units.mol, "Ce2O3": (140.12 * 2 + 16 * 3) * units.g / units.mol, "Pr2O3": (140.91 * 2 + 16 * 3) * units.g / units.mol, "Nd2O3": (144.24 * 2 + 16 * 3) * units.g / units.mol, "Sm2O3": (150.36 * 2 + 16 * 3) * units.g / units.mol, "Gd2O3": (157.25 * 2 + 16 * 3) * units.g / units.mol, "Dy2O3": (162.5 * 2 + 16 * 3) * units.g / units.mol, } REE_mass_frac = { "Y2O3": 88.906 * 2 / (88.906 * 2 + 16 * 3), "La2O3": 138.91 * 2 / (138.91 * 2 + 16 * 3), "Ce2O3": 140.12 * 2 / (140.12 * 2 + 16 * 3), "Pr2O3": 140.91 * 2 / (140.91 * 2 + 16 * 3), "Nd2O3": 144.24 * 2 / (144.24 * 2 + 16 * 3), "Sm2O3": 150.36 * 2 / (150.36 * 2 + 16 * 3), "Gd2O3": 157.25 * 2 / (157.25 * 2 + 16 * 3), "Dy2O3": 162.5 * 2 / (162.5 * 2 + 16 * 3), } # Total mass basis yield calculation product = value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Y"] * molar_mass["Y2O3"] * REE_mass_frac["Y2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "La"] * molar_mass["La2O3"] * REE_mass_frac["La2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Ce"] * molar_mass["Ce2O3"] * REE_mass_frac["Ce2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Pr"] * molar_mass["Pr2O3"] * REE_mass_frac["Pr2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Nd"] * molar_mass["Nd2O3"] * REE_mass_frac["Nd2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Sm"] * molar_mass["Sm2O3"] * REE_mass_frac["Sm2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Gd"] * molar_mass["Gd2O3"] * REE_mass_frac["Gd2O3"], to_units=units.kg / units.hr, ) + units.convert( m.fs.roaster.flow_mol_comp_product[0, "Dy"] * molar_mass["Dy2O3"] * REE_mass_frac["Dy2O3"], to_units=units.kg / units.hr, ) ) print(f"REE product mass flow is {product} kg/hr") feed_REE = sum( value( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, molecule] ) * REE_frac for molecule, REE_frac in REE_mass_frac.items() ) print(f"REE feed mass flow is {feed_REE} kg/hr") REE_recovery = 100 * product / feed_REE print(f"Total REE recovery is {REE_recovery} %") product_purity = ( 100 * product / value( units.convert( m.fs.roaster.flow_mass_product[0], to_units=units.kg / units.hr ) ) ) print(f"Product purity is {product_purity} % REE") # Individual elemental recoveries total_al_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Al"] * molar_mass["Al2O3"] * metal_mass_frac["Al2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Al2O3"] * metal_mass_frac["Al2O3"], to_units=units.kg / units.hr, ) ) ) total_fe_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Fe"] * molar_mass["Fe2O3"] * metal_mass_frac["Fe2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Fe2O3"] * metal_mass_frac["Fe2O3"], to_units=units.kg / units.hr, ) ) ) total_ca_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Ca"] * molar_mass["CaO"] * metal_mass_frac["CaO"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "CaO"] * metal_mass_frac["CaO"], to_units=units.kg / units.hr, ) ) ) total_sc_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Sc"] * molar_mass["Sc2O3"] * metal_mass_frac["Sc2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Sc2O3"] * metal_mass_frac["Sc2O3"], to_units=units.kg / units.hr, ) ) ) total_yt_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Y"] * molar_mass["Y2O3"] * metal_mass_frac["Y2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Y2O3"] * metal_mass_frac["Y2O3"], to_units=units.kg / units.hr, ) ) ) total_la_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "La"] * molar_mass["La2O3"] * metal_mass_frac["La2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "La2O3"] * metal_mass_frac["La2O3"], to_units=units.kg / units.hr, ) ) ) total_ce_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Ce"] * molar_mass["Ce2O3"] * metal_mass_frac["Ce2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Ce2O3"] * metal_mass_frac["Ce2O3"], to_units=units.kg / units.hr, ) ) ) total_pr_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Pr"] * molar_mass["Pr2O3"] * metal_mass_frac["Pr2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Pr2O3"] * metal_mass_frac["Pr2O3"], to_units=units.kg / units.hr, ) ) ) total_nd_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Nd"] * molar_mass["Nd2O3"] * metal_mass_frac["Nd2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Nd2O3"] * metal_mass_frac["Nd2O3"], to_units=units.kg / units.hr, ) ) ) total_sm_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Sm"] * molar_mass["Sm2O3"] * metal_mass_frac["Sm2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Sm2O3"] * metal_mass_frac["Sm2O3"], to_units=units.kg / units.hr, ) ) ) total_gd_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Gd"] * molar_mass["Gd2O3"] * metal_mass_frac["Gd2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Gd2O3"] * metal_mass_frac["Gd2O3"], to_units=units.kg / units.hr, ) ) ) total_dy_recovery = 100 * value( units.convert( m.fs.roaster.flow_mol_comp_product[0, "Dy"] * molar_mass["Dy2O3"] * metal_mass_frac["Dy2O3"], to_units=units.kg / units.hr, ) / ( units.convert( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, "Dy2O3"] * metal_mass_frac["Dy2O3"], to_units=units.kg / units.hr, ) ) ) al_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Al"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Al2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Al2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) ca_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Ca"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["CaO"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "CaO"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) ce_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Ce"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Ce2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Ce2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) dy_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Dy"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Dy2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Dy2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) fe_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Fe"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Fe2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Fe2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) gd_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Gd"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Gd2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Gd2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) la_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "La"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["La2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "La2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) nd_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Nd"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Nd2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Nd2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) pr_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Pr"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Pr2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Pr2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) sc_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Sc"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Sc2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Sc2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) sm_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Sm"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Sm2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Sm2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) yt_recovery = value( units.convert( m.fs.sl_sep1.recovered_liquid_outlet.conc_mass_comp[0, "Y"] * m.fs.sl_sep1.recovered_liquid_outlet.flow_vol[0], to_units=units.kg / units.hr, ) / ( units.convert( metal_mass_frac["Y2O3"] * m.fs.leach_solid_feed.outlet.mass_frac_comp[0, "Y2O3"] * m.fs.leach_solid_feed.outlet.flow_mass[0], to_units=units.kg / units.hr, ) ) * 100 ) print(f"\nLeaching Aluminum recovery is {al_recovery} %") print(f"Total aluminum recovery is {total_al_recovery} %") print(f"\nLeaching Calcium recovery is {ca_recovery} %") print(f"Total Calcium recovery is {total_ca_recovery} %") print(f"\nLeaching Cerium recovery is {ce_recovery} %") print(f"Total Cerium recovery is {total_ce_recovery} %") print(f"\nLeaching Dysprosium recovery is {dy_recovery} %") print(f"Total Dysprosium recovery is {total_dy_recovery} %") print(f"\nLeaching Iron recovery is {fe_recovery} %") print(f"Total Iron recovery is {total_fe_recovery} %") print(f"\nLeaching Gadolinium recovery is {gd_recovery} %") print(f"Total Gadolinium recovery is {total_gd_recovery} %") print(f"\nLeaching Lanthanum recovery is {la_recovery} %") print(f"Total Lanthanum recovery is {total_la_recovery} %") print(f"\nLeaching Neodymium recovery is {nd_recovery} %") print(f"Total Neodymium recovery is {total_nd_recovery} %") print(f"\nLeaching Praseodymium recovery is {pr_recovery} %") print(f"Total Praseodymium recovery is {total_pr_recovery} %") print(f"\nLeaching Scandium recovery is {sc_recovery} %") print(f"Total Scandium recovery is {total_sc_recovery} %") print(f"\nLeaching Samarium recovery is {sm_recovery} %") print(f"Total Samarium recovery is {total_sm_recovery} %") print(f"\nLeaching Yttrium recovery is {yt_recovery} %") print(f"Total Yttrium recovery is {total_yt_recovery} %")
[docs] def add_costing(m): """ Set the costing parameters for each unit model. Args: m: pyomo model """ # TODO: Costing is preliminary until more unit model costing metrics can be verified # TODO: Should ideally define balance-of-plant equipment in the flowsheet and attach costing to it, # eliminating the need to create UnitModelBlocks in the costing m.fs.costing = QGESSCosting() CE_index_year = "UKy_2019" # Leaching costs # 4.2 is UKy Leaching - Polyethylene Tanks L_pe_tanks_accounts = ["4.2"] m.fs.L_pe_tanks = UnitModelBlock() m.fs.L_pe_tanks.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_pe_tanks_accounts, "scaled_param": m.fs.leach.volume[0, 1], "source": 1, "n_equip": 3, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 4.3 is UKy Leaching - Tank Mixer L_tank_mixer_accounts = ["4.3"] m.fs.L_tank_mixers = UnitModelBlock() m.fs.L_tank_mixers.power = Var(initialize=4.74, units=units.hp) m.fs.L_tank_mixers.power.fix() m.fs.L_tank_mixers.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_tank_mixer_accounts, "scaled_param": m.fs.L_tank_mixers.power, "source": 1, "n_equip": 3, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 4.4 is UKy Leaching - Process Pump L_pump_accounts = ["4.4"] m.fs.L_pump = UnitModelBlock() m.fs.L_pump.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_pump_accounts, "scaled_param": m.fs.leach_liquid_feed.flow_vol[0], "source": 1, "n_equip": 3, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 4.5 is UKy Leaching - Thickener L_thickener_accounts = ["4.5"] m.fs.L_thickener = UnitModelBlock() m.fs.L_thickener.area = Var(initialize=225.90, units=units.ft**2) m.fs.L_thickener.area.fix() m.fs.L_thickener.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_thickener_accounts, "scaled_param": m.fs.L_thickener.area, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 4.6 is UKy Leaching - Solid Waste Filter Press L_filter_press_accounts = ["4.6"] m.fs.L_filter_press = UnitModelBlock() m.fs.L_filter_press.volume = Var(initialize=36.00, units=units.ft**3) m.fs.L_filter_press.volume.fix() m.fs.L_filter_press.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_filter_press_accounts, "scaled_param": m.fs.L_filter_press.volume, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 4.8 is UKy Leaching - Solution Heater L_solution_heater_accounts = ["4.8"] m.fs.L_solution_heater = UnitModelBlock() m.fs.L_solution_heater.duty = Var(initialize=0.24, units=units.MBTU / units.hr) m.fs.L_solution_heater.duty.fix() m.fs.L_solution_heater.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": L_solution_heater_accounts, "scaled_param": m.fs.L_solution_heater.duty, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # Solvent extraction costs # 5.1 is UKy Rougher Solvent Extraction - Polyethylene Tanks RSX_pe_tanks_accounts = ["5.1"] m.fs.RSX_pe_tanks = UnitModelBlock() m.fs.RSX_pe_tanks.capacity = Var(initialize=35.136, units=units.gal) m.fs.RSX_pe_tanks.capacity.fix() m.fs.RSX_pe_tanks.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": RSX_pe_tanks_accounts, "scaled_param": m.fs.RSX_pe_tanks.capacity, "source": 1, "n_equip": 6, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 5.2 is UKy Rougher Solvent Extraction - Tank Mixer RSX_tank_mixer_accounts = ["5.2"] m.fs.RSX_tank_mixers = UnitModelBlock() m.fs.RSX_tank_mixers.power = Var(initialize=2.0, units=units.hp) m.fs.RSX_tank_mixers.power.fix() m.fs.RSX_tank_mixers.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": RSX_tank_mixer_accounts, "scaled_param": m.fs.RSX_tank_mixers.power, "source": 1, "n_equip": 2, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 5.3 is UKy Rougher Solvent Extraction - Process Pump RSX_pump_accounts = ["5.3"] m.fs.RSX_pump = UnitModelBlock() m.fs.RSX_pump.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": RSX_pump_accounts, "scaled_param": m.fs.solex_rougher_load.mscontactor.aqueous_inlet.flow_vol[ 0 ], "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 5.4 is UKy Rougher Solvent Extraction - Mixer Settler RSX_mixer_settler_accounts = ["5.4"] m.fs.RSX_mixer_settler = UnitModelBlock() m.fs.RSX_mixer_settler.volume = Var(initialize=61.107, units=units.gal) m.fs.RSX_mixer_settler.volume.fix() m.fs.RSX_mixer_settler.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": RSX_mixer_settler_accounts, "scaled_param": m.fs.RSX_mixer_settler.volume, "source": 1, "n_equip": 6, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 6.1 is UKy Cleaner Solvent Extraction - Polyethylene Tanks CSX_pe_tanks_accounts = ["6.1"] m.fs.CSX_pe_tanks = UnitModelBlock() m.fs.CSX_pe_tanks.capacity = Var(initialize=14.05, units=units.gal) m.fs.CSX_pe_tanks.capacity.fix() m.fs.CSX_pe_tanks.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": CSX_pe_tanks_accounts, "scaled_param": m.fs.CSX_pe_tanks.capacity, "source": 1, "n_equip": 5, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 6.2 is UKy Cleaner Solvent Extraction - Tank Mixer CSX_tank_mixer_accounts = ["6.2"] m.fs.CSX_tank_mixers = UnitModelBlock() m.fs.CSX_tank_mixers.power = Var(initialize=0.08, units=units.hp) m.fs.CSX_tank_mixers.power.fix() m.fs.CSX_tank_mixers.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": CSX_tank_mixer_accounts, "scaled_param": m.fs.CSX_tank_mixers.power, "source": 1, "n_equip": 2, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 6.3 is UKy Cleaner Solvent Extraction - Process Pump CSX_pump_accounts = ["6.3"] m.fs.CSX_pump = UnitModelBlock() m.fs.CSX_pump.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": CSX_pump_accounts, "scaled_param": m.fs.solex_cleaner_load.mscontactor.aqueous_inlet.flow_vol[ 0 ], "source": 1, "n_equip": 3, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 6.4 is UKy Cleaner Solvent Extraction - Mixer Settler CSX_mixer_settler_accounts = ["6.4"] m.fs.CSX_mixer_settler = UnitModelBlock() m.fs.CSX_mixer_settler.volume = Var(initialize=24.44, units=units.gal) m.fs.CSX_mixer_settler.volume.fix() m.fs.CSX_mixer_settler.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": CSX_mixer_settler_accounts, "scaled_param": m.fs.CSX_mixer_settler.volume, "source": 1, "n_equip": 6, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # Precipitation costs # 9.2 is UKy Rare Earth Element Precipitation - Polyethylene Tanks reep_pe_tanks_accounts = ["9.2"] m.fs.reep_pe_tanks = UnitModelBlock() m.fs.reep_pe_tanks.capacity = Var(initialize=15.04, units=units.gal) m.fs.reep_pe_tanks.capacity.fix() m.fs.reep_pe_tanks.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": reep_pe_tanks_accounts, "scaled_param": m.fs.reep_pe_tanks.capacity, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 9.3 is UKy Rare Earth Element Precipitation - Tank Mixer reep_tank_mixer_accounts = ["9.3"] m.fs.reep_tank_mixers = UnitModelBlock() m.fs.reep_tank_mixers.power = Var(initialize=0.61, units=units.hp) m.fs.reep_tank_mixers.power.fix() m.fs.reep_tank_mixers.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": reep_tank_mixer_accounts, "scaled_param": m.fs.reep_tank_mixers.power, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 9.4 is UKy Rare Earth Element Precipitation - Process Pump reep_pump_accounts = ["9.4"] m.fs.reep_pump = UnitModelBlock() m.fs.reep_pump.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": reep_pump_accounts, "scaled_param": m.fs.precipitator.aqueous_inlet.flow_vol[0], "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 9.5 is UKy Rare Earth Element Precipitation - Filter Press reep_filter_press_accounts = ["9.5"] m.fs.reep_filter_press = UnitModelBlock() m.fs.reep_filter_press.volume = Var(initialize=0.405, units=units.ft**3) m.fs.reep_filter_press.volume.fix() m.fs.reep_filter_press.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": reep_filter_press_accounts, "scaled_param": m.fs.reep_filter_press.volume, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 9.8 is UKy Rare Earth Element Precipitation - Roaster reep_roaster_accounts = ["9.8"] m.fs.reep_roaster = UnitModelBlock() m.fs.reep_roaster.duty = Var(initialize=0.035, units=units.MBTU / units.hr) m.fs.reep_roaster.duty.fix() m.fs.reep_roaster.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": reep_roaster_accounts, "scaled_param": m.fs.reep_roaster.duty, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # Roasting costs # 3.1 is UKy Roasting - Storage Bins R_storage_bins_accounts = ["3.1"] m.fs.R_storage_bins = UnitModelBlock() m.fs.R_storage_bins.capacity = Var(initialize=10.0, units=units.ton) m.fs.R_storage_bins.capacity.fix() m.fs.R_storage_bins.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_storage_bins_accounts, "scaled_param": m.fs.R_storage_bins.capacity, "source": 1, "n_equip": 2, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 3.2 is UKy Roasting - Conveyors R_conveyors_accounts = ["3.2"] m.fs.R_conveyors = UnitModelBlock() m.fs.R_conveyors.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_conveyors_accounts, "scaled_param": m.fs.roaster.flow_mass_product[0], "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 3.3 is UKy Roasting - Roaster R_roaster_accounts = ["3.3"] m.fs.R_roaster = UnitModelBlock() m.fs.R_roaster.duty = Var(initialize=73.7, units=units.MBTU / units.hr) m.fs.R_roaster.duty.fix() m.fs.R_roaster.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_roaster_accounts, "scaled_param": m.fs.R_roaster.duty, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 3.4 is UKy Roasting - Gas Scrubber R_gas_scrubber_accounts = ["3.4"] m.fs.R_gas_scrubber = UnitModelBlock() m.fs.R_gas_scrubber.gas_rate = Var(initialize=11.500, units=units.ft**3 / units.min) m.fs.R_gas_scrubber.gas_rate.fix() m.fs.R_gas_scrubber.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_gas_scrubber_accounts, "scaled_param": m.fs.R_gas_scrubber.gas_rate, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 3.5 is UKy Roasting - Spray Chamber Quencher (7000-60000 ft**3/min) R_spray_chamber_quencher_accounts = ["3.5"] m.fs.R_spray_chamber_quencher = UnitModelBlock() m.fs.R_spray_chamber_quencher.gas_rate = Var( initialize=11.500, units=units.ft**3 / units.min ) m.fs.R_spray_chamber_quencher.gas_rate.fix() m.fs.R_spray_chamber_quencher.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_spray_chamber_quencher_accounts, "scaled_param": m.fs.R_spray_chamber_quencher.gas_rate, "source": 1, "n_equip": 3, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) # 3.7 is UKy Roasting - Chiller R_chiller_accounts = ["3.7"] m.fs.R_chiller = UnitModelBlock() m.fs.R_chiller.duty = Var(initialize=13.1, units=units.MBTU / units.hr) m.fs.R_chiller.duty.fix() m.fs.R_chiller.costing = UnitModelCostingBlock( flowsheet_costing_block=m.fs.costing, costing_method=QGESSCostingData.get_REE_costing, costing_method_arguments={ "cost_accounts": R_chiller_accounts, "scaled_param": m.fs.R_chiller.duty, "source": 1, "n_equip": 1, "scale_down_parallel_equip": False, "CE_index_year": CE_index_year, }, ) REE_mass_frac = { "Y2O3": 88.906 * 2 / (88.906 * 2 + 16 * 3), "La2O3": 138.91 * 2 / (138.91 * 2 + 16 * 3), "Ce2O3": 140.12 * 2 / (140.12 * 2 + 16 * 3), "Pr2O3": 140.91 * 2 / (140.91 * 2 + 16 * 3), "Nd2O3": 144.24 * 2 / (144.24 * 2 + 16 * 3), "Sm2O3": 150.36 * 2 / (150.36 * 2 + 16 * 3), "Gd2O3": 157.25 * 2 / (157.25 * 2 + 16 * 3), "Dy2O3": 162.5 * 2 / (162.5 * 2 + 16 * 3), } feed_REE = sum( m.fs.leach_solid_feed.flow_mass[0] * m.fs.leach_solid_feed.mass_frac_comp[0, molecule] * REE_frac for molecule, REE_frac in REE_mass_frac.items() ) m.fs.feed_input = Var(initialize=0.025, units=units.ton / units.hr) m.fs.feed_input_constraint = Constraint( expr=m.fs.feed_input == units.convert( m.fs.leach_solid_feed.flow_mass[0], to_units=units.ton / units.hr ) ) m.fs.feed_grade = Var(initialize=318.015, units=units.ppm) m.fs.feed_grade_constraint = Constraint( expr=m.fs.feed_grade == units.convert( feed_REE / m.fs.leach_solid_feed.flow_mass[0], to_units=units.ppm ) ) hours_per_shift = 8 shifts_per_day = 3 operating_days_per_year = 336 # for convenience m.fs.annual_operating_hours = Param( initialize=hours_per_shift * shifts_per_day * operating_days_per_year, mutable=True, units=units.hours / units.a, ) m.fs.recovery_rate_per_year = Var(initialize=13.306, units=units.kg / units.yr) m.fs.recovery_rate_per_year_constraint = Constraint( expr=m.fs.recovery_rate_per_year == units.convert( m.fs.roaster.flow_mass_product[0] * m.fs.annual_operating_hours, to_units=units.kg / units.yr, ) ) # the land cost is the lease cost, or refining cost of REO produced m.fs.land_cost = Expression( expr=0.303736 * 1e-6 * getattr(units, "MUSD_" + CE_index_year) / units.ton * units.convert(m.fs.feed_input, to_units=units.ton / units.hr) * hours_per_shift * units.hr * shifts_per_day * units.day**-1 * operating_days_per_year * units.day ) m.fs.solid_waste = Var(m.fs.time, initialize=0.0245, units=units.ton / units.hr) m.fs.solid_waste_constraint = Constraint( expr=m.fs.solid_waste[0] == units.convert( m.fs.leach_filter_cake.flow_mass[0], to_units=units.ton / units.hr ) ) m.fs.precipitate = Var( m.fs.time, initialize=0, units=units.ton / units.hr ) # non-hazardous precipitate m.fs.dust_and_volatiles = Var( m.fs.time, initialize=9.5e-8, units=units.ton / units.hr ) m.fs.dust_and_volatiles_constraint = Constraint( expr=m.fs.dust_and_volatiles[0] == units.convert(m.fs.roaster.flow_mass_dust[0], to_units=units.ton / units.hr) ) m.fs.power = Var(m.fs.time, initialize=7, units=units.hp) m.fs.power_constraint = Constraint( expr=m.fs.power[0] == units.convert( m.fs.reep_tank_mixers.power + m.fs.CSX_tank_mixers.power + m.fs.RSX_tank_mixers.power + m.fs.L_tank_mixers.power, to_units=units.hp, ) ) resources = [ "nonhazardous_solid_waste", "nonhazardous_precipitate_waste", "dust_and_volatiles", "power", ] rates = [ m.fs.solid_waste, m.fs.precipitate, m.fs.dust_and_volatiles, m.fs.power, ] # define product flowrates REO_molar_mass = { "Y2O3": 88.906 * 2 + 16 * 3, "La2O3": 138.91 * 2 + 16 * 3, "Ce2O3": 140.12 * 2 + 16 * 3, "Pr2O3": 140.91 * 2 + 16 * 3, "Nd2O3": 144.24 * 2 + 16 * 3, "Sm2O3": 150.36 * 2 + 16 * 3, "Gd2O3": 157.25 * 2 + 16 * 3, "Dy2O3": 162.5 * 2 + 16 * 3, "Sc2O3": 44.96 * 2 + 16 * 3, } m.fs.Ce_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Ce"] * REO_molar_mass["Ce2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product cerium oxide mass flow", ) m.fs.Dy_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Dy"] * REO_molar_mass["Dy2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product dysprosium oxide mass flow", ) m.fs.Gd_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Gd"] * REO_molar_mass["Gd2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product gadolinium oxide mass flow", ) m.fs.La_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "La"] * REO_molar_mass["La2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product lanthanum oxide mass flow", ) m.fs.Nd_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Nd"] * REO_molar_mass["Nd2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product neodymium oxide mass flow", ) m.fs.Pr_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Pr"] * REO_molar_mass["Pr2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product praseodymium oxide mass flow", ) m.fs.Sc_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Sc"] * REO_molar_mass["Sc2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product scandium oxide mass flow", ) m.fs.Sm_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Sm"] * REO_molar_mass["Sm2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product samarium oxide mass flow", ) m.fs.Y_product = Param( default=units.convert( m.fs.roaster.flow_mol_comp_product[0, "Y"] * REO_molar_mass["Y2O3"] * units.g / units.mol, to_units=units.kg / units.hr, ), units=units.kg / units.hr, mutable=True, doc="Product yttrium oxide mass flow", ) pure_product_output_rates = {} mixed_product_output_rates = { "CeO2": m.fs.Ce_product, "Sc2O3": m.fs.Sc_product, "Y2O3": m.fs.Y_product, "La2O3": m.fs.La_product, "Nd2O3": m.fs.Nd_product, "Pr6O11": m.fs.Pr_product, "Sm2O3": m.fs.Sm_product, "Gd2O3": m.fs.Gd_product, "Dy2O3": m.fs.Dy_product, } m.fs.costing.build_process_costs( # arguments related to installation costs piping_materials_and_labor_percentage=20, electrical_materials_and_labor_percentage=20, instrumentation_percentage=8, plants_services_percentage=10, process_buildings_percentage=40, auxiliary_buildings_percentage=15, site_improvements_percentage=10, equipment_installation_percentage=17, field_expenses_percentage=12, project_management_and_construction_percentage=30, process_contingency_percentage=15, # argument related to Fixed OM costs labor_types=[ "skilled", "unskilled", "supervisor", "maintenance", "technician", "engineer", ], labor_rate=[24.98, 19.08, 30.39, 22.73, 21.97, 45.85], # USD/hr labor_burden=25, # % fringe benefits operators_per_shift=[4, 9, 2, 2, 2, 3], hours_per_shift=hours_per_shift, shifts_per_day=shifts_per_day, operating_days_per_year=operating_days_per_year, pure_product_output_rates=pure_product_output_rates, mixed_product_output_rates=mixed_product_output_rates, mixed_product_sale_price_realization_factor=0.65, # 65% price realization for mixed products # arguments related to total owners costs land_cost=m.fs.land_cost, resources=resources, rates=rates, fixed_OM=True, variable_OM=True, feed_input=m.fs.feed_input, efficiency=0.80, # power usage efficiency, or fixed motor/distribution efficiency waste=[ "nonhazardous_solid_waste", "nonhazardous_precipitate_waste", "dust_and_volatiles", ], recovery_rate_per_year=m.fs.recovery_rate_per_year, CE_index_year=CE_index_year, ) # define reagent fill costs as an other plant cost so framework adds this to TPC calculation m.fs.costing.other_plant_costs.unfix() m.fs.costing.other_plant_costs_eq = Constraint( expr=( m.fs.costing.other_plant_costs == units.convert( 1218.073 * units.USD_2016 # Rougher Solvent Extraction + 48.723 * units.USD_2016 # Cleaner Solvent Extraction + 182.711 * units.USD_2016, # Solvent Extraction Wash and Saponification to_units=getattr(units, "MUSD_" + CE_index_year), ) ) ) # fix costing vars that shouldn't change m.fs.precipitate.fix() # Initialize costing QGESSCostingData.costing_initialization(m.fs.costing) QGESSCostingData.initialize_fixed_OM_costs(m.fs.costing) QGESSCostingData.initialize_variable_OM_costs(m.fs.costing) # Solve costing solver = get_solver() solver.solve(m, tee=True) return m
[docs] def display_costing(m): """ Print the key costing results. Args: m: pyomo model """ QGESSCostingData.report(m.fs.costing) m.fs.costing.variable_operating_costs.display() QGESSCostingData.display_bare_erected_costs(m.fs.costing) QGESSCostingData.display_flowsheet_cost(m.fs.costing)
if __name__ == "__main__": m, results = main()