Source code for reconstruction.ecoli.dataclasses.process.metabolism

"""
SimulationData for metabolism process

TODO:

- improved estimate of ILE/LEU abundance or some external data point
- implement L1-norm minimization for AA concentrations
- find concentration for PI[c]
- add (d)NTP byproduct concentrations
"""
# mypy: disable-error-code=attr-defined

from copy import copy
import itertools
import re
from typing import (
    Any,
    Callable,
    cast,
    Iterable,
    Optional,
    Set,
    Tuple,
    Union,
    TYPE_CHECKING,
)
from unum import Unum

from numba import njit
import numpy as np
import numpy.typing as npt
import sympy as sp
from sympy.parsing.sympy_parser import parse_expr

from ecoli.library.schema import bulk_name_to_idx, counts
from reconstruction.ecoli.dataclasses.getter_functions import (
    UNDEFINED_COMPARTMENT_IDS_TO_ABBREVS,
)
from reconstruction.ecoli.knowledge_base_raw import KnowledgeBaseEcoli

if TYPE_CHECKING:
    from reconstruction.ecoli.simulation_data import SimulationDataEcoli
from reconstruction.ecoli.dataclasses.constants import Constants
from wholecell.utils import units

KINETIC_CONSTRAINT_CONC_UNITS = units.umol / units.L
K_CAT_UNITS = 1 / units.s
"""Units for k :sub:`cat` values"""
METABOLITE_CONCENTRATION_UNITS = units.mol / units.L
"""Units for metabolite concentrations"""
DRY_MASS_UNITS = units.fg
"""Units for dry mass"""

USE_ALL_CONSTRAINTS = False  # False will remove defined constraints from objective

REVERSE_TAG = " (reverse)"
REVERSE_REACTION_ID = "{{}}{}".format(REVERSE_TAG)
ENZYME_REACTION_ID = "{}__{}"

VERBOSE = False


[docs] class InvalidReactionDirectionError(Exception): pass
[docs] class Metabolism(object): """Metabolism Args: raw_data: Raw data object sim_data: Simulation data object Attributes: solver (str): solver ID, should match a value in modular_fba.py, set by :py:meth:`~._set_solver_values` kinetic_objective_weight (float): weighting for the kinetic objective, 1-weighting for the homeostatic objective, set by :py:meth:`~._set_solver_values` kinetic_objective_weight_in_range (float): weighting for deviations from the kinetic target within min and max ranges, set by :py:meth:`~._set_solver_values` secretion_penalty_coeff (float): penalty on secretion fluxes, set by :py:meth:`~._set_solver_values` metabolite_charge (dict[str, int]): mapping of metabolite IDs to charge, set by :py:meth:`~._add_metabolite_charge` concentration_updates conc_dict (dict): nutrients_to_internal_conc (dict[str, dict[str, Unum]]): kinetic_constraint_reactions: kinetic_constraint_enzymes: kinetic_constraint_substrates: _kcats: _saturations: _enzymes: constraint_is_kcat_only: _compiled_enzymes: _compiled_saturation: reaction_stoich: maintenance_reaction: reaction_catalysts: catalyst_ids: reactions_with_catalyst: catalysis_matrix_I: catalysis_matrix_J: catalysis_matrix_V: use_all_constraints: constraints_to_disable: base_reaction_ids: reaction_id_to_base_reaction_id: amino_acid_export_kms: transport_reactions (list[str]): transport reaction IDs in the metabolic network (includes reverse reactions and reactions with kinetic constraints), set by :py:meth:`~._build_transport_reactions` aa_synthesis_pathways (dict[str, dict]): data for allosteric inhibition of amino acid pathways indexed by amino acid ID with location tag and nested dictionary with the following keys:: {'enzymes' (str): limiting/regulated enzyme ID in synthesis pathway with location tag, 'kcat_data' (units.Unum): kcat associated with enzyme reaction with units of 1/time, 'ki' (Tuple[units.Unum, units.Unum]]): lower and upper limits of KI associated with enzyme reaction with units of mol/volume} Set by :py:meth:`~._build_amino_acid_pathways` KI_aa_synthesis (numpy.ndarray[float]): KI for each AA for synthesis portion of supply (in units of :py:data:`~.METABOLITE_CONCENTRATION_UNITS`), set by :py:meth:`~.set_phenomological_supply_constants` KM_aa_export (numpy.ndarray[float]): KM for each AA for export portion of supply (in units of :py:data:`~.METABOLITE_CONCENTRATION_UNITS`), set by :py:meth:`~.set_phenomological_supply_constants` fraction_supply_rate (float): fraction of AA supply that comes from a base synthesis rate, set by :py:meth:`~.set_phenomological_supply_constants` fraction_import_rate (numpy.ndarray[float]): fraction of AA supply that comes from AA import if nutrients are present, set by :py:meth:`~.set_phenomological_supply_constants` ppgpp_synthesis_reaction (str): reaction ID for ppGpp synthesis (catalyzed by RelA and SpoT), set by :py:meth:`~._build_ppgpp_reactions` ppgpp_degradation_reaction (str): reaction ID for ppGpp degradation (catalyzed by SpoT), set by :py:meth:`~._build_ppgpp_reactions` ppgpp_reaction_names (list[str]): names of reaction involved in ppGpp, set by :py:meth:`~._build_ppgpp_reactions` ppgpp_reaction_metabolites (list[str]): names of metabolites in ppGpp reactions, set by :py:meth:`~._build_ppgpp_reactions` ppgpp_reaction_stoich (numpy.ndarray[int]): 2D array with metabolites on rows and reactions on columns containing the stoichiometric coefficient, set by :py:meth:`~._build_ppgpp_reactions` aa_to_exporters (dict[str, list]): dictonary that maps aa to transporters involved in export reactions. Set by :py:meth:`~.set_mechanistic_export_constants`. aa_to_exporters_matrix (numpy.ndarray[int]): correlation matrix. Columns correspond to exporting enzymes and rows to amino acids. Set by :py:meth:`~.set_mechanistic_export_constants`. aa_exporter_names (numpy.ndarray[str]): names of all exporters. Set by :py:meth:`~.set_mechanistic_export_constants`. aa_export_kms (numpy.ndarray[float]): kms corresponding to generic transport/enzyme reactions for each AA in concentration units. Set by :py:meth:`~.set_mechanistic_export_constants`. export_kcats_per_aa (numpy.ndarray[float]): kcats corresponding to generic export reactions for each AA. Units in counts/second. Set by :py:meth:`~.set_mechanistic_export_constants` and :py:meth:`~.set_mechanistic_export_constants`. aa_to_importers (dict[str, list]): dictonary that maps aa to transporters involved in import reactions. Set by :py:meth:`~.set_mechanistic_uptake_constants`. aa_to_importers_matrix (numpy.ndarray[int]): correlation matrix. Columns correspond to importing enzymes and rows to amino acids. Set by :py:meth:`~.set_mechanistic_export_constants`. aa_importer_names (numpy.ndarray[str]): names of all importers. Set by :py:meth:`~.set_mechanistic_export_constants`. import_kcats_per_aa (numpy.ndarray[float]): kcats corresponding to generic import reactions for each AA. Units in counts/second. Set by :py:meth:`~.set_mechanistic_export_constants`. aa_enzymes (numpy.ndarray[str]): enzyme ID with location tag for each enzyme that can catalyze an amino acid pathway with :py:attr:`~.enzyme_to_amino_acid` mapping these to each amino acid. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_kcats_fwd (numpy.ndarray[float]): forward kcat value for each synthesis pathway in units of :py:data:`~.K_CAT_UNITS`, ordered by amino acid molecule group. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_kcats_rev (numpy.ndarray[float]): reverse kcat value for each synthesis pathway in units of :py:data:`~.K_CAT_UNITS`, ordered by amino acid molecule group. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_kis (numpy.ndarray[float]): KI value for each synthesis pathway in units of :py:data:`~.METABOLITE_CONCENTRATION_UNITS`, ordered by amino acid molecule group. Will be inf if there is no inhibitory control. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_upstream_kms (list[list[float]]): KM value associated with the amino acid that feeds into each synthesis pathway in units of :py:data:`~.METABOLITE_CONCENTRATION_UNITS`, ordered by amino acid molecule group. Will be 0 if there is no upstream amino acid considered. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_reverse_kms (numpy.ndarray[float]): KM value associated with the amino acid in each synthesis pathway in units of :py:data:`~.METABOLITE_CONCENTRATION_UNITS`, ordered by amino acid molecule group. Will be inf if the synthesis pathway is not reversible. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_upstream_mapping (numpy.ndarray[int]): index of the upstream amino acid that feeds into each synthesis pathway, ordered by amino acid molecule group. Set by :py:meth:`~.set_mechanistic_supply_constants`. enzyme_to_amino_acid (numpy.ndarray[float]): relationship mapping from aa_enzymes to amino acids (n enzymes, m amino acids). Will contain a 1 if the enzyme associated with the row can catalyze the pathway for the amino acid associated with the column. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_forward_stoich (numpy.ndarray[float]): relationship mapping from upstream amino acids to downstream amino acids (n upstream, m downstream). Will contain a -1 if the amino acid associated with the row is required for synthesis of the amino acid associated with the column. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_reverse_stoich (numpy.ndarray[float]): relationship mapping from upstream amino acids to downstream amino acids (n downstream, m upstream). Will contain a -1 if the amino acid associated with the row is produced through a reverse reaction from the amino acid associated with the column. Set by :py:meth:`~.set_mechanistic_supply_constants`. aa_import_kis (numpy.ndarray[float]): inhibition constants for amino acid import based on the internal amino acid concentration specific_import_rates (numpy.ndarray[float]): import rates expected in rich media conditions for each amino acid normalized by dry cell mass in units of :py:data:`~.K_CAT_UNITS` / :py:data:`~.DRY_MASS_UNITS`, ordered by amino acid molecule group. Set by :py:meth:`~.set_mechanistic_supply_constants`. max_specific_import_rates (numpy.ndarray[float]): max import rates for each amino acid without including internal concentration inhibition normalized by dry cell mass in units of :py:data:`~.K_CAT_UNITS` / :py:data:`~.DRY_MASS_UNITS`, ordered by amino acid molecule group. Set by :py:meth:`~.set_mechanistic_supply_constants`. """ def __init__(self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli"): self._set_solver_values(sim_data.constants) self._build_biomass(raw_data, sim_data) self._build_metabolism(raw_data, sim_data) self._build_ppgpp_reactions(raw_data, sim_data) self._build_transport_reactions(raw_data, sim_data) self._build_amino_acid_pathways(raw_data, sim_data) self._add_metabolite_charge(raw_data)
[docs] def _add_metabolite_charge(self, raw_data: KnowledgeBaseEcoli): """ Compiles all metabolite charges. Args: raw_data: Raw data object Attributes set: - :py:attr:`~.metabolite_charge` """ self.metabolite_charge = {} for met in raw_data.metabolites: self.metabolite_charge[met["id"]] = met["molecular_charge"]
[docs] def _set_solver_values(self, constants: Constants): """ Sets values to be used in the FBA solver. Attributes set: - :py:attr:`~.solver` - :py:attr:`~.kinetic_objective_weight` - :py:attr:`~.secretion_penalty_coeff` """ self.solver = "glpk-linear" if "linear" in self.solver: self.kinetic_objective_weight = ( constants.metabolism_kinetic_objective_weight_linear ) else: self.kinetic_objective_weight = ( constants.metabolism_kinetic_objective_weight_quadratic ) self.kinetic_objective_weight_in_range = ( constants.metabolism_kinetic_objective_weight_in_range ) self.secretion_penalty_coeff = constants.secretion_penalty_coeff
[docs] def _build_biomass( self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ): """ Calculates metabolite concentration targets. Args: raw_data: Raw data object sim_data: Simulation data object Attributes set: - :py:attr:`~.concentration_updates` - :py:attr:`~.conc_dict` - :py:attr:`~.nutrients_to_internal_conc` """ wildtypeIDs = set(entry["molecule id"] for entry in raw_data.biomass) # Create vector of metabolite target concentrations # Since the data only covers certain metabolites, we need to rationally # expand the dataset to include the other molecules in the biomass # function. # First, load in metabolites that do have concentrations, then assign # compartments according to those given in the biomass objective. Or, # if there is no compartment, assign it to the cytoplasm. concentration_sources = [ "Park Concentration", "Lempp Concentration", "Kochanowski Concentration", "Sander Concentration", ] excluded = { "Park Concentration": { "GLT", # Steady state concentration reached with tRNA charging is much lower than Park "THR", # Attenuation needs concentration to be lower to match validation data "VAL", # Synthesis pathway kcat needs concentration to be lower and closer to KI }, "Lempp Concentration": { "ATP", # TF binding does not solve with average concentration "VAL", # Synthesis pathway kcat needs concentration to be lower and closer to KI }, "Kochanowski Concentration": { "ATP", # TF binding does not solve with average concentration }, "Sander Concentration": { "GLT", # Steady state concentration reached with tRNA charging is much lower than Sander }, } metaboliteIDs = [] metaboliteConcentrations = [] wildtypeIDtoCompartment = { wildtypeID[:-3]: wildtypeID[-3:] for wildtypeID in wildtypeIDs } # this assumes biomass reaction components only exist in a single compartment for row in raw_data.metabolite_concentrations: metabolite_id = row["Metabolite"] if not sim_data.getter.is_valid_molecule(metabolite_id): if VERBOSE: print( "Metabolite concentration for unknown molecule: {}".format( metabolite_id ) ) continue # Use average of both sources # TODO (Travis): geometric mean? conc = np.nanmean( [ row[source].asNumber(METABOLITE_CONCENTRATION_UNITS) for source in concentration_sources if metabolite_id not in excluded.get(source, set()) ] ) # Check that a value was in the datasets being used if not np.isfinite(conc): if VERBOSE: print( "No concentration in active datasets for {}".format( metabolite_id ) ) continue if metabolite_id in wildtypeIDtoCompartment: metaboliteIDs.append( metabolite_id + wildtypeIDtoCompartment[metabolite_id] ) else: metaboliteIDs.append(metabolite_id + "[c]") metaboliteConcentrations.append(conc) # CYS/SEL: concentration based on other amino acids aaConcentrations = [] for aaIndex, aaID in enumerate(sim_data.amino_acid_code_to_id_ordered.values()): if aaID in metaboliteIDs: metIndex = metaboliteIDs.index(aaID) aaConcentrations.append(metaboliteConcentrations[metIndex]) aaSmallestConc = min(aaConcentrations) metaboliteIDs.append("CYS[c]") metaboliteConcentrations.append(aaSmallestConc) metaboliteIDs.append("L-SELENOCYSTEINE[c]") metaboliteConcentrations.append(aaSmallestConc) # DGTP: set to smallest of all other DNTP concentrations dntpConcentrations = [] for dntpIndex, dntpID in enumerate(sim_data.molecule_groups.dntps): if dntpID in metaboliteIDs: metIndex = metaboliteIDs.index(dntpID) dntpConcentrations.append(metaboliteConcentrations[metIndex]) dntpSmallestConc = min(dntpConcentrations) metaboliteIDs.append("DGTP[c]") metaboliteConcentrations.append(dntpSmallestConc) # H: from reported pH hydrogenConcentration = 10 ** (-sim_data.constants.pH) metaboliteIDs.append(sim_data.molecule_ids.proton) metaboliteConcentrations.append(hydrogenConcentration) # PPI ppi_conc = sim_data.constants.ppi_concentration.asNumber( METABOLITE_CONCENTRATION_UNITS ) metaboliteIDs.append(sim_data.molecule_ids.ppi) metaboliteConcentrations.append(ppi_conc) metaboliteIDs.append("Pi[c]") metaboliteConcentrations.append(ppi_conc) # include metabolites that are part of biomass d = sim_data.mass.getBiomassAsConcentrations(sim_data.doubling_time) for key, value in d.items(): metaboliteIDs.append(key) metaboliteConcentrations.append( value.asNumber(METABOLITE_CONCENTRATION_UNITS) ) # Load relative metabolite changes relative_changes: dict[str, dict[str, float]] = {} for row in raw_data.relative_metabolite_concentrations: met = row["Metabolite"] met_id = met + wildtypeIDtoCompartment.get(met, "[c]") # AA concentrations are determined through charging if met_id in sim_data.molecule_groups.amino_acids: continue # Get relative metabolite change in each media condition for col, value in row.items(): # Skip the ID column and minimal column (only has values of 1) # or skip invalid values if col == "Metabolite" or col == "minimal" or not np.isfinite(value): continue if col not in relative_changes: relative_changes[col] = {} relative_changes[col][met_id] = value ## Add manually curated values for other media for ( media, data, ) in sim_data.adjustments.relative_metabolite_concentrations_changes.items(): if media not in relative_changes: relative_changes[media] = {} for met, change in data.items(): if met not in relative_changes[media]: relative_changes[media][met] = change # save concentrations as class variables unique_ids, counts = np.unique(metaboliteIDs, return_counts=True) if np.any(counts > 1): raise ValueError( "Multiple concentrations for metabolite(s): {}".format( ", ".join(unique_ids[counts > 1]) ) ) # TODO (Travis): only pass raw_data and sim_data and create functions to load absolute and relative concentrations conc_dict = dict( zip( metaboliteIDs, METABOLITE_CONCENTRATION_UNITS * np.array(metaboliteConcentrations), ) ) all_metabolite_ids = {met["id"] for met in raw_data.metabolites} linked_metabolites = self._build_linked_metabolites(raw_data, conc_dict) self.concentration_updates = ConcentrationUpdates( conc_dict, relative_changes, raw_data.equilibrium_reactions, sim_data.external_state.exchange_dict, all_metabolite_ids, linked_metabolites, ) self.conc_dict = self.concentration_updates.concentrations_based_on_nutrients( "minimal" ) self.nutrients_to_internal_conc = {} self.nutrients_to_internal_conc["minimal"] = self.conc_dict.copy()
[docs] def _build_linked_metabolites( self, raw_data: KnowledgeBaseEcoli, conc_dict: dict[str, Unum] ) -> dict[str, dict[str, Any]]: """ Calculates ratio between linked metabolites to keep it constant throughout a simulation. Args: raw_data: Raw data object conc_dict: Mapping of metabolite IDs to homeostatic concentrations calculated by :py:meth:`~_build_biomass` Returns: Mapping from a linked metabolite to its lead metabolite and concentration ratio to be maintained:: {'lead' (str): metabolite to link the concentration to, 'ratio' (float): ratio to multiply the lead concentration by} """ linked_metabolites = {} for row in raw_data.linked_metabolites: lead = row["Lead metabolite"] linked = row["Linked metabolite"] ratio = units.strip_empty_units(conc_dict[linked] / conc_dict[lead]) linked_metabolites[linked] = {"lead": lead, "ratio": ratio} return linked_metabolites
[docs] def _build_metabolism( self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ): """ Build the matrices/vectors for metabolism (FBA) Reads in and stores reaction and kinetic constraint information Args: raw_data: Raw data object sim_data: Simulation data object Attributes set: - :py:attr:`~.kinetic_constraint_reactions` - :py:attr:`~.kinetic_constraint_enzymes` - :py:attr:`~.kinetic_constraint_substrates` - :py:attr:`~._kcats` - :py:attr:`~._saturations` - :py:attr:`~._enzymes` - :py:attr:`~.constraint_is_kcat_only` - :py:attr:`~._compiled_enzymes` - :py:attr:`~._compiled_saturation` - :py:attr:`~.reaction_stoich` - :py:attr:`~.maintenance_reaction` - :py:attr:`~.reaction_catalysts` - :py:attr:`~.catalyst_ids` - :py:attr:`~.reactions_with_catalyst` - :py:attr:`~.catalysis_matrix_I` - :py:attr:`~.catalysis_matrix_J` - :py:attr:`~.catalysis_matrix_V` - :py:attr:`~.use_all_constraints` - :py:attr:`~.constraints_to_disable` - :py:attr:`~.base_reaction_ids` - :py:attr:`~.reaction_id_to_base_reaction_id` - :py:attr:`~.amino_acid_export_kms` """ ( base_rxn_ids, reaction_stoich, reversible_reactions, catalysts, rxn_id_to_base_rxn_id, ) = self.extract_reactions(raw_data, sim_data) # Load kinetic reaction constraints from raw_data known_metabolites = set(self.conc_dict) raw_constraints = self.extract_kinetic_constraints( raw_data, sim_data, stoich=reaction_stoich, catalysts=catalysts, known_metabolites=known_metabolites, ) # Make modifications from kinetics data ( constraints, reaction_stoich, catalysts, reversible_reactions, rxn_id_to_base_rxn_id, ) = self._replace_enzyme_reactions( raw_constraints, reaction_stoich, catalysts, reversible_reactions, rxn_id_to_base_rxn_id, ) # Create symbolic kinetic equations ( self.kinetic_constraint_reactions, self.kinetic_constraint_enzymes, self.kinetic_constraint_substrates, self._kcats, self._saturations, self._enzymes, self.constraint_is_kcat_only, ) = self._lambdify_constraints(constraints) self._compiled_enzymes: Optional[ Callable[[npt.NDArray[np.float64]], npt.NDArray[np.float64]] ] = None self._compiled_saturation: Optional[ Callable[[npt.NDArray[np.float64]], npt.NDArray[np.float64]] ] = None # TODO: move this to a sim_data analysis script if VERBOSE: print("\nSummary of included metabolism kinetics:") print( "Reactions with kinetics: {}".format( len(self.kinetic_constraint_reactions) ) ) print( "Enzymes with kinetics: {}".format(len(self.kinetic_constraint_enzymes)) ) print( "Metabolites in kinetics: {}".format( len(self.kinetic_constraint_substrates) ) ) print( "Number of kcat values: {}".format( len([k for c in constraints.values() for k in c["kcat"]]) ) ) print( "Number of saturation terms: {}".format( len([s for c in constraints.values() for s in c["saturation"]]) ) ) # Verify no substrates with unknown concentrations have been added unknown = { m for m in self.kinetic_constraint_substrates if m not in known_metabolites } if unknown: raise ValueError( "Unknown concentration for {}. Need to remove" " kinetics saturation term.".format(", ".join(unknown)) ) # Extract data reactions_with_catalyst = sorted(catalysts) catalyst_ids = sorted({c for all_cat in catalysts.values() for c in all_cat}) # Create catalysis matrix (to be used in the simulation) catalysisMatrixI = [] catalysisMatrixJ = [] catalysisMatrixV = [] for row, reaction in enumerate(reactions_with_catalyst): for catalyst in catalysts[reaction]: col = catalyst_ids.index(catalyst) catalysisMatrixI.append(row) catalysisMatrixJ.append(col) catalysisMatrixV.append(1) catalysisMatrixI = np.array(catalysisMatrixI) catalysisMatrixJ = np.array(catalysisMatrixJ) catalysisMatrixV = np.array(catalysisMatrixV) # Properties for FBA reconstruction self.reaction_stoich = reaction_stoich # TODO (ggsun): add this as a raw .tsv file self.maintenance_reaction = { "ATP[c]": -1, "WATER[c]": -1, "ADP[c]": +1, "Pi[c]": +1, "PROTON[c]": +1, } # Properties for catalysis matrix (to set hard bounds) self.reaction_catalysts = catalysts self.catalyst_ids = catalyst_ids self.reactions_with_catalyst = reactions_with_catalyst self.catalysis_matrix_I = catalysisMatrixI self.catalysis_matrix_J = catalysisMatrixJ self.catalysis_matrix_V = catalysisMatrixV # Properties for setting flux targets self.use_all_constraints = USE_ALL_CONSTRAINTS self.constraints_to_disable = [ rxn["disabled reaction"] for rxn in raw_data.disabled_kinetic_reactions ] # Properties for conversion of fluxes to those for base reaction IDs self.base_reaction_ids = base_rxn_ids self.reaction_id_to_base_reaction_id = rxn_id_to_base_rxn_id self.amino_acid_export_kms = raw_data.amino_acid_export_kms
[docs] def _build_ppgpp_reactions( self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ): """ Creates structures for ppGpp reactions for use in polypeptide_elongation. Args: raw_data: Raw data object sim_data: Simulation data object Attributes set: - :py:attr:`~.ppgpp_synthesis_reaction` - :py:attr:`~.ppgpp_degradation_reaction` - :py:attr:`~.ppgpp_reaction_names` - :py:attr:`~.ppgpp_reaction_metabolites` - :py:attr:`~.ppgpp_reaction_stoich` """ self.ppgpp_synthesis_reaction = "GDPPYPHOSKIN-RXN" self.ppgpp_degradation_reaction = "PPGPPSYN-RXN" self.ppgpp_reaction_names = [ self.ppgpp_synthesis_reaction, self.ppgpp_degradation_reaction, ] self.ppgpp_reaction_metabolites = [] # Indices (i: metabolite, j: reaction) and values (v: stoichiometry) # for sparse reaction matrix metabolite_indices: dict[str, int] = {} new_index = 0 rxn_i = [] rxn_j = [] rxn_v = [] # Record sparse indices in the matrix for j, rxn in enumerate(self.ppgpp_reaction_names): for met, stoich in self.reaction_stoich[rxn].items(): idx = metabolite_indices.get(met, new_index) if idx == new_index: metabolite_indices[met] = new_index self.ppgpp_reaction_metabolites.append(met) new_index += 1 rxn_i.append(idx) rxn_j.append(j) rxn_v.append(stoich) # Assemble matrix based on indices # new_index is number of metabolites, j+1 is number of reactions self.ppgpp_reaction_stoich = np.zeros((new_index, j + 1), dtype=np.int32) self.ppgpp_reaction_stoich[rxn_i, rxn_j] = rxn_v
[docs] def _build_transport_reactions( self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ): """ Creates list of transport reactions that are included in the reaction network. Args: raw_data: Raw data object sim_data: Simulation data object Attributes set: - :py:attr:`~.transport_reactions` """ transport_reactions = [ rxn_id for rxn_id, stoich in self.reaction_stoich.items() if self._is_transport_rxn(stoich) ] self.transport_reactions = transport_reactions
[docs] def _build_amino_acid_pathways( self, raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ): """ Creates mapping between enzymes and amino acid pathways with allosteric inhibition feedback from the amino acid. Args: raw_data: Raw data object sim_data: Simulation data object Attributes set: - :py:attr:`~.aa_synthesis_pathways` """ self.aa_synthesis_pathways = {} cytoplasm_tag = "[c]" for row in raw_data.amino_acid_pathways: data: dict[str, Any] = {} data["enzymes"] = [ e + sim_data.getter.get_compartment_tag(e) for e in row["Enzymes"] ] data["reverse enzymes"] = [ e + sim_data.getter.get_compartment_tag(e) for e in row["Reverse enzymes"] ] data["kcat_data"] = 0 / units.s if units.isnan(row["kcat"]) else row["kcat"] if units.isnan(row["KI, lower bound"]) or units.isnan( row["KI, lower bound"] ): data["ki"] = None else: data["ki"] = (row["KI, lower bound"], row["KI, upper bound"]) data["upstream"] = { k + cytoplasm_tag: v for k, v in row["Upstream amino acids"].items() } data["reverse"] = { k + cytoplasm_tag: v for k, v in row["Reverse amino acids"].items() } data["km, upstream"] = { k + cytoplasm_tag: v for k, v in row["KM, upstream"].items() } data["km, reverse"] = row["KM, reverse"] data["km, degradation"] = ( np.inf * units.mol / units.L if units.isnan(row["KM, degradation"]) else row["KM, degradation"] ) data["downstream"] = { k + cytoplasm_tag: v for k, v in row["Downstream amino acids"].items() } self.aa_synthesis_pathways[row["Amino acid"] + cytoplasm_tag] = data self.aa_synthesis_pathway_adjustments: dict[str, dict[str, float]] = {} for row in raw_data.adjustments.amino_acid_pathways: # Read data from row aa = row["Amino acid"] + cytoplasm_tag parameter = row["Parameter"] factor = row["Factor"] # Store adjustments to be used later adjustments = self.aa_synthesis_pathway_adjustments.get(aa, {}) adjustments[parameter] = factor self.aa_synthesis_pathway_adjustments[aa] = adjustments self.amino_acid_uptake_rates = {} for row in raw_data.amino_acid_uptake_rates: rates = {} rates["uptake"] = row["Uptake"] rates["LB"] = row["Uptake, LB"] rates["UB"] = row["Uptake, UB"] self.amino_acid_uptake_rates[row["Amino acid"]] = rates
[docs] def get_kinetic_constraints(self, enzymes: Unum, substrates: Unum) -> Unum: """ Allows for dynamic code generation for kinetic constraint calculation for use in Metabolism process. Inputs should be unitless but the order of magnitude should match the kinetics parameters (umol/L/s). If trying to pickle sim_data object after function has been called, _compiled_enzymes and _compiled_saturation might not be able to be pickled. See __getstate__(), __setstate__() comments on PR 111 to address. Returns np.array of floats of the kinetic constraint target for each reaction with kinetic parameters Args: enzymes: concentrations of enzymes associated with kinetic constraints (mol / volume units) substrates: concentrations of substrates associated with kinetic constraints (mol / volume units) Returns: Array of dimensions (n reactions, 3) where each row contains the min, mean and max kinetic constraints for each reaction with kinetic constraints (mol / volume / time units) """ if self._compiled_enzymes is None: self._compiled_enzymes = eval("lambda e: {}".format(self._enzymes)) if self._compiled_saturation is None: self._compiled_saturation = eval("lambda s: {}".format(self._saturations)) # Strip units from args enzs = enzymes.asNumber(KINETIC_CONSTRAINT_CONC_UNITS) subs = substrates.asNumber(KINETIC_CONSTRAINT_CONC_UNITS) capacity = np.array(self._compiled_enzymes(enzs))[:, None] * self._kcats saturation = np.array( [[min(v), sum(v) / len(v), max(v)] for v in self._compiled_saturation(subs)] ) return KINETIC_CONSTRAINT_CONC_UNITS * K_CAT_UNITS * capacity * saturation
[docs] def exchange_constraints( self, exchangeIDs, coefficient, targetUnits, media_id, unconstrained, constrained, concModificationsBasedOnCondition=None, ): """ Called during Metabolism process Returns the homeostatic objective concentrations based on the current nutrients Returns levels for external molecules available to exchange based on the current nutrients """ newObjective = self.concentration_updates.concentrations_based_on_nutrients( imports=unconstrained.union(constrained), media_id=media_id, conversion_units=targetUnits, ) if concModificationsBasedOnCondition is not None: newObjective.update(concModificationsBasedOnCondition) externalMoleculeLevels = np.zeros(len(exchangeIDs), np.float64) for index, moleculeID in enumerate(exchangeIDs): if moleculeID in unconstrained: externalMoleculeLevels[index] = np.inf elif moleculeID in constrained: externalMoleculeLevels[index] = ( constrained[moleculeID] * coefficient ).asNumber(targetUnits) else: externalMoleculeLevels[index] = 0.0 return externalMoleculeLevels, newObjective
[docs] def set_phenomological_supply_constants(self, sim_data: "SimulationDataEcoli"): """ Sets constants to determine amino acid supply during translation. Used with aa_supply_scaling() during simulations but supply can alternatively be determined mechanistically. This approach may require manually adjusting constants (fraction_supply_inhibited and fraction_supply_exported) but has less variability related to gene expression and regulation. Args: sim_data: simulation data Attributes set: - :py:attr:`~.KI_aa_synthesis` - :py:attr:`~.KM_aa_export` - :py:attr:`~.fraction_supply_rate` - :py:attr:`~.fraction_import_rate` Assumptions: - Each internal amino acid concentration in 'minimal_plus_amino_acids' media is not lower than in 'minimal' media TODO (Travis): Better handling of concentration assumption """ aa_ids = sim_data.molecule_groups.amino_acids conc = self.concentration_updates.concentrations_based_on_nutrients aa_conc_basal = np.array( [ conc(media_id="minimal")[aa].asNumber(METABOLITE_CONCENTRATION_UNITS) for aa in aa_ids ] ) aa_conc_aa_media = np.array( [ conc(media_id="minimal_plus_amino_acids")[aa].asNumber( METABOLITE_CONCENTRATION_UNITS ) for aa in aa_ids ] ) # Lower concentrations might produce strange rates (excess supply or # negative import when present externally) and constants so raise # to double check the implementation if not np.all(aa_conc_basal <= aa_conc_aa_media): aas = np.array(aa_ids)[np.where(aa_conc_basal > aa_conc_aa_media)] raise ValueError( "Check that amino acid concentrations should be lower in amino acid media for {}".format( aas ) ) f_inhibited = sim_data.constants.fraction_supply_inhibited f_exported = sim_data.constants.fraction_supply_exported # Assumed units of METABOLITE_CONCENTRATION_UNITS for KI and KM self.KI_aa_synthesis = f_inhibited * aa_conc_basal / (1 - f_inhibited) self.KM_aa_export = (1 / f_exported - 1) * aa_conc_aa_media self.fraction_supply_rate = ( 1 - f_inhibited + aa_conc_basal / (self.KM_aa_export + aa_conc_basal) ) self.fraction_import_rate = 1 - ( self.fraction_supply_rate + 1 / (1 + aa_conc_aa_media / self.KI_aa_synthesis) - f_exported )
[docs] def aa_supply_scaling( self, aa_conc: Unum, aa_present: Unum ) -> npt.NDArray[np.float64]: """ Called during polypeptide_elongation process Determine amino acid supply rate scaling based on current amino acid concentrations. Args: aa_conc: internal concentration for each amino acid (ndarray[float]) aa_present: whether each amino acid is in the external environment or not (ndarray[bool]) Returns: Scaling for the supply of each amino acid with higher supply rate if >1, lower supply rate if <1 """ aa_conc = aa_conc.asNumber(METABOLITE_CONCENTRATION_UNITS) aa_supply = self.fraction_supply_rate aa_import = aa_present * self.fraction_import_rate aa_synthesis = 1 / (1 + aa_conc / self.KI_aa_synthesis) aa_export = aa_conc / (self.KM_aa_export + aa_conc) supply_scaling = aa_supply + aa_import + aa_synthesis - aa_export return supply_scaling
[docs] def get_aa_to_transporters_mapping_data( self, sim_data: "SimulationDataEcoli", export: bool = False ) -> tuple[dict[str, list], npt.NDArray[np.float64], npt.NDArray[np.str_]]: """ Creates a dictionary that maps amino acids with their transporters. Based on this dictionary, it creates a correlation matrix with rows as AA and columns as transporters. Args: sim_data: simulation data export: if True, the parameters calculated are for mechanistic export instead of uptake Returns: 3-element tuple containing - aa_to_transporters: dictonary that maps aa to transporters involved in transport reactions - aa_to_transporters_matrix: correlation matrix. Columns correspond to transporter enzymes and rows to amino acids - aa_transporters_names: names of all transporters """ def matches_direction(direction): if export: return direction < 0 else: return direction > 0 # Mapping aminoacids to their transporters # CYS does not have any uptake reaction, so we initialize the dict with it to ensure # the presence of the 21 AAs # TODO (Santiago): Reversible reactions? aa_to_transporters: dict[str, list[str]] = {"CYS[c]": []} for reaction in self.transport_reactions: for aa in sim_data.molecule_groups.amino_acids: if aa in self.reaction_stoich[reaction] and matches_direction( self.reaction_stoich[reaction][aa] ): if aa not in aa_to_transporters: aa_to_transporters[aa] = [] aa_to_transporters[aa] += self.reaction_catalysts.get(reaction, []) aa_to_transporters = { aa: aa_to_transporters[aa] for aa in sim_data.molecule_groups.amino_acids } c = 0 transporters_to_idx = {} for aa, transporters in aa_to_transporters.items(): for transporter in transporters: if transporter not in transporters_to_idx: transporters_to_idx[transporter] = c c += 1 aa_to_transporters_matrix = [[0]] * len(aa_to_transporters) for i, trnspts in enumerate(aa_to_transporters.values()): temp = [0] * len(transporters_to_idx) for tr in trnspts: temp[transporters_to_idx[tr]] = 1 aa_to_transporters_matrix[i] = temp aa_transporters_names = list(transporters_to_idx.keys()) return ( aa_to_transporters, np.array(aa_to_transporters_matrix), np.array(aa_transporters_names), )
[docs] def set_mechanistic_export_constants( self, sim_data: "SimulationDataEcoli", cell_specs: dict[str, dict], basal_container: np.ndarray, ): """ Calls get_aa_to_transporters_mapping_data() for AA export, which calculates the total amount of export transporter counts per AA. Kcats are calculated using the same exchange rates as for uptake and transporter counts. Missing KMs are calculated based on present KMs. This is done by calculating the average factor for KMs compared to estimated concentration (av_factor = sum(KM / concentration) / n_aa_with_kms). ** KM = av_factor * concentration Args: sim_data: simulation data cell_specs: mapping from condition to calculated cell properties basal_container: average initial bulk molecule counts in the basal condition (structured Numpy array, see :ref:`bulk`) Attributes set: - :py:attr:`~.aa_to_exporters` - :py:attr:`~.aa_to_exporters_matrix` - :py:attr:`~.aa_exporter_names` - :py:attr:`~.aa_export_kms` - :py:attr:`~.export_kcats_per_aa` """ self.aa_to_exporters, self.aa_to_exporters_matrix, self.aa_exporter_names = ( self.get_aa_to_transporters_mapping_data(sim_data, export=True) ) aa_names = sim_data.molecule_groups.amino_acids aa_idx = bulk_name_to_idx(aa_names, basal_container["id"]) counts_to_molar = ( sim_data.constants.cell_density / cell_specs["basal"]["avgCellDryMassInit"] ) / sim_data.constants.n_avogadro aa_conc = { aa: counts * counts_to_molar for aa, counts in zip(aa_names, counts(basal_container, aa_idx)) } aa_with_km = {} # Calculate average factor to estimate missing KMs coeff_estimate_kms = 0.0 for export_kms in self.amino_acid_export_kms: aa_with_km[export_kms["Amino Acid"]] = export_kms["KM"] coef_per_aa = 0 for km in export_kms["KM"].values(): coef_per_aa += ( km.asUnit(METABOLITE_CONCENTRATION_UNITS) / aa_conc[export_kms["Amino Acid"]] ) coeff_estimate_kms += coef_per_aa / len(export_kms["KM"]) coeff_estimate_kms = coeff_estimate_kms / len(self.amino_acid_export_kms) # Calculate estimated KMs for each AA single_kms = {} for aa in aa_names: if aa in aa_with_km: single_kms[aa] = np.mean(list(aa_with_km[aa].values())) else: single_kms[aa] = coeff_estimate_kms * aa_conc[aa] self.aa_export_kms = np.array( [single_kms[aa].asNumber(METABOLITE_CONCENTRATION_UNITS) for aa in aa_names] )
[docs] def set_mechanistic_uptake_constants( self, sim_data: "SimulationDataEcoli", cell_specs: dict[str, dict], with_aa_container: np.ndarray, ): """ Based on the matrix calculated in get_aa_to_transporters_mapping_data(), we calculate the total amount of transporter counts per AA. Args: sim_data: simulation data cell_specs: mapping from condition to calculated cell properties with_aa_container: average initial bulk molecule counts in the ``with_aa`` condition (structured Numpy array, see :ref:`bulk`) Attributes set: - :py:attr:`~.aa_to_importers` - :py:attr:`~.aa_to_importers_matrix` - :py:attr:`~.aa_importer_names` - :py:attr:`~.import_kcats_per_aa` - :py:attr:`~.export_kcats_per_aa` TODO: - Include external amino acid concentrations and KM values """ aa_names = sim_data.molecule_groups.amino_acids aa_idx = bulk_name_to_idx(aa_names, with_aa_container["id"]) counts_to_molar = ( sim_data.constants.cell_density / cell_specs["with_aa"]["avgCellDryMassInit"] ) / sim_data.constants.n_avogadro aa_conc = counts(with_aa_container, aa_idx) * counts_to_molar.asNumber( METABOLITE_CONCENTRATION_UNITS ) exchange_rates = self.specific_import_rates * cell_specs["with_aa"][ "avgCellDryMassInit" ].asNumber(units.fg) self.aa_to_importers, self.aa_to_importers_matrix, self.aa_importer_names = ( self.get_aa_to_transporters_mapping_data(sim_data) ) aa_importer_idx = bulk_name_to_idx( self.aa_importer_names, with_aa_container["id"] ) importer_counts = counts(with_aa_container, aa_importer_idx) aa_exporter_idx = bulk_name_to_idx( self.aa_exporter_names, with_aa_container["id"] ) exporter_counts = counts(with_aa_container, aa_exporter_idx) counts_per_aa_import = self.aa_to_importers_matrix.dot(importer_counts) counts_per_aa_export = self.aa_to_exporters_matrix.dot(exporter_counts) # Solve for the two unknown kcats with the calculated net exchange rate # in rich media conditions and the assumption that import and export # rates are equal at the export KM based on how the export KM values # were curated. # Import will decrease and export will increase with higher amino acids # for stable amino acid concentrations. import_saturation_in_rich = 1 / (1 + aa_conc / self.aa_import_kis) export_saturation_in_rich = 1 / (1 + self.aa_export_kms / aa_conc) import_saturation_at_km = 1 / (1 + self.aa_export_kms / self.aa_import_kis) export_saturation_at_km = 0.5 import_capacity_at_km = counts_per_aa_import * import_saturation_at_km export_capacity_at_km = counts_per_aa_export * export_saturation_at_km with np.errstate(divide="ignore", invalid="ignore"): import_vs_export_kcat = export_capacity_at_km / import_capacity_at_km kcat_export = exchange_rates / ( import_vs_export_kcat * counts_per_aa_import * import_saturation_in_rich - counts_per_aa_export * export_saturation_in_rich ) kcat_export[~np.isfinite(kcat_export)] = 0 kcat_import = import_vs_export_kcat * kcat_export kcat_import[~np.isfinite(kcat_import)] = 0 if np.any(kcat_export < 0) or np.any(kcat_import < 0): raise ValueError( "Could not solve for positive transport kcat." " Check assumptions or amino acid concentrations compared to KMs." ) self.export_kcats_per_aa = kcat_export self.import_kcats_per_aa = kcat_import
[docs] def set_mechanistic_supply_constants( self, sim_data: "SimulationDataEcoli", cell_specs: dict[str, dict], basal_container: np.ndarray, with_aa_container: np.ndarray, ): """ Sets constants to determine amino acid supply during translation. Used with amino_acid_synthesis() and amino_acid_import() during simulations but supply can alternatively be determined phenomologically. This approach is more detailed and should better respond to environmental changes and perturbations but has more variability related to gene expression and regulation. Args: sim_data: simulation data cell_specs: mapping from condition to calculated cell properties basal_container: average initial bulk molecule counts in the basal condition (structured Numpy array, see :ref:`bulk`) with_aa_container: average initial bulk molecule counts in the ``with_aa`` condition Sets class attributes: - :py:attr:`~.aa_enzymes` - :py:attr:`~.aa_kcats_fwd` - :py:attr:`~.aa_kcats_rev` - :py:attr:`~.aa_kis` - :py:attr:`~.aa_upstream_kms` - :py:attr:`~.aa_reverse_kms` - :py:attr:`~.aa_upstream_mapping` - :py:attr:`~.enzyme_to_amino_acid` - :py:attr:`~.aa_forward_stoich` - :py:attr:`~.aa_reverse_stoich` - :py:attr:`~.aa_import_kis` - :py:attr:`~.specific_import_rates` - :py:attr:`~.max_specific_import_rates` Assumptions: - Only one reaction is limiting in an amino acid pathway (typically the first and one with KI) and the kcat for forward or reverse directions will apply to all enzymes that can catalyze that step - kcat for reverse and degradation reactions is the same (each amino acid only has reverse or degradation at this point but that could change with modifications to the amino_acid_pathways flat file) TODO: - Search for new kcat/KM values in literature or use metabolism_kinetics.tsv - Consider multiple reaction steps - Include mulitple amino acid inhibition on importers (currently amino acids only inhibit their own import but some transporters import multiple amino acids and will be inhibited by all of the amino acids for the import of each amino acid) """ aa_ids = sim_data.molecule_groups.amino_acids n_aas = len(aa_ids) self.aa_to_index = {aa: i for i, aa in enumerate(aa_ids)} conc = self.concentration_updates.concentrations_based_on_nutrients # Measured data used as targets for calculations measured_uptake_rates = {} fwd_kcat_targets = {} for aa in aa_ids: aa_no_tag = aa[:-3] measured_uptake_rates[aa_no_tag] = ( self.amino_acid_uptake_rates[aa_no_tag]["uptake"].asNumber( units.mmol / units.g / units.h ) if aa_no_tag in self.amino_acid_uptake_rates else 0 ) if ( kcat := self.aa_synthesis_pathways[aa]["kcat_data"].asNumber( K_CAT_UNITS ) ) > 0: fwd_kcat_targets[aa_no_tag] = kcat default_fwd_target = np.mean(list(fwd_kcat_targets.values())) # Allosteric inhibition constants to match required supply rate basal_rates = ( sim_data.translation_supply_rate["minimal"] * cell_specs["basal"]["avgCellDryMassInit"] * sim_data.constants.n_avogadro ).asNumber(K_CAT_UNITS) with_aa_rates = ( sim_data.translation_supply_rate["minimal_plus_amino_acids"] * cell_specs["with_aa"]["avgCellDryMassInit"] * sim_data.constants.n_avogadro ).asNumber(K_CAT_UNITS) basal_supply_mapping = dict(zip(aa_ids, basal_rates)) with_aa_supply_mapping = dict(zip(aa_ids, with_aa_rates)) aa_enzymes = [] enzyme_to_aa_fwd = [] enzyme_to_aa_rev = [] aa_kcats_fwd = {} aa_kcats_rev = {} aa_kis = {} upstream_aas_for_km = {} aa_upstream_kms = {} aa_reverse_kms = {} aa_degradation_kms = {} fwd_rates: dict[str, float] = {} rev_rates: dict[str, float] = {} deg_rates: dict[str, float] = {} calculated_uptake_rates = {} minimal_conc = conc("minimal") with_aa_conc = conc("minimal_plus_amino_acids") # Get order of amino acids to calculate parameters for to ensure that # parameters that are dependent on other amino acids are run after # those calculations have completed self.aa_forward_stoich = np.eye(n_aas) self.aa_reverse_stoich = np.eye(n_aas) dependencies: dict[str, set[str]] = {} for aa in aa_ids: for downstream_aa in self.aa_synthesis_pathways[aa]["downstream"]: if units.isfinite( self.aa_synthesis_pathways[downstream_aa]["km, degradation"] ): dependencies.setdefault(aa, set()).add(downstream_aa) # Convert individual supply calculations to overall supply based on dependencies # via dot product (self.aa_forward_stoich @ supply) for upstream_aa, stoich in self.aa_synthesis_pathways[aa][ "upstream" ].items(): self.aa_forward_stoich[ self.aa_to_index[upstream_aa], self.aa_to_index[aa] ] = -stoich dependencies.setdefault(upstream_aa, set()).add(aa) for reverse_aa, stoich in self.aa_synthesis_pathways[aa]["reverse"].items(): self.aa_reverse_stoich[ self.aa_to_index[reverse_aa], self.aa_to_index[aa] ] = -stoich dependencies.setdefault(reverse_aa, set()).add(aa) ordered_aa_ids: list[str] = [] for _ in aa_ids: # limit number of iterations number of amino acids in case there are cyclic links for aa in sorted(set(aa_ids) - set(ordered_aa_ids)): for downstream_aa in dependencies.get(aa, set()): if downstream_aa not in ordered_aa_ids: break else: ordered_aa_ids.append(aa) if len(ordered_aa_ids) != n_aas: raise RuntimeError( "Could not determine amino acid order to calculate dependencies first." " Make sure there are no cyclical pathways for amino acids that can degrade." ) for amino_acid in ordered_aa_ids: data = self.aa_synthesis_pathways[amino_acid] fwd_enzymes = data["enzymes"] fwd_enzymes_basal_idx = bulk_name_to_idx(fwd_enzymes, basal_container["id"]) fwd_enzymes_basal = counts(basal_container, fwd_enzymes_basal_idx).sum() fwd_enzymes_with_aa_idx = bulk_name_to_idx( fwd_enzymes, with_aa_container["id"] ) fwd_enzymes_with_aa = counts( with_aa_container, fwd_enzymes_with_aa_idx ).sum() rev_enzymes = data["reverse enzymes"] rev_enzymes_basal_idx = bulk_name_to_idx(rev_enzymes, basal_container["id"]) rev_enzymes_basal = counts(basal_container, rev_enzymes_basal_idx).sum() rev_enzymes_with_aa_idx = bulk_name_to_idx( rev_enzymes, with_aa_container["id"] ) rev_enzymes_with_aa = counts( with_aa_container, rev_enzymes_with_aa_idx ).sum() aa_conc_basal = minimal_conc[amino_acid] aa_conc_with_aa = with_aa_conc[amino_acid] if data["ki"] is None: ki = np.inf * units.mol / units.L else: # Get largest dynamic range possible given the range of measured KIs lower_limit, upper_limit = data["ki"] if aa_conc_basal < lower_limit: ki = lower_limit elif aa_conc_basal > upper_limit: ki = upper_limit else: ki = aa_conc_basal upstream_aa = [aa for aa in data["upstream"]] km_conc_basal = METABOLITE_CONCENTRATION_UNITS * np.array( [ minimal_conc[aa].asNumber(METABOLITE_CONCENTRATION_UNITS) for aa in upstream_aa ] ) km_conc_with_aa = METABOLITE_CONCENTRATION_UNITS * np.array( [ with_aa_conc[aa].asNumber(METABOLITE_CONCENTRATION_UNITS) for aa in upstream_aa ] ) kms_upstream = data["km, upstream"] kms = METABOLITE_CONCENTRATION_UNITS * np.array( [ kms_upstream.get(aa, minimal_conc[aa]).asNumber( METABOLITE_CONCENTRATION_UNITS ) for aa in upstream_aa ] ) # TODO: better way to fill in this missing data if data["reverse"]: if units.isnan(data["km, reverse"]): km_reverse = ( minimal_conc[amino_acid] * 10 ) # TODO: better way to fill in this missing data else: km_reverse = data["km, reverse"] # TODO: remove this if separating reverse and deg enzymes elif ( data["reverse enzymes"] and not units.isfinite(data["km, degradation"]) and amino_acid != "L-SELENOCYSTEINE[c]" ): km_reverse = ( minimal_conc[amino_acid] * 10 ) # TODO: better way to fill in this missing data else: km_reverse = np.inf * units.mol / units.L km_degradation = data["km, degradation"] # Make required adjustments in order to get positive kcats and import rates for parameter, factor in self.aa_synthesis_pathway_adjustments.get( amino_acid, {} ).items(): if parameter == "ki": ki *= factor elif parameter == "km_degradation": km_degradation *= factor elif parameter == "km_reverse": km_reverse *= factor elif parameter == "kms": kms *= factor else: raise ValueError( f"Unexpected parameter adjustment ({parameter}) for {amino_acid}." ) if units.isfinite(km_reverse) and units.isfinite(km_degradation): raise ValueError( "Currently cannot have a reverse and degradation KM for amino acid" f" synthesis pathways ({amino_acid}). Consider a method to solve for separate" " kcats and implement matching saturation calculation in amino_acid_synthesis()" " and calc_kcats()" ) def calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, kms, km_reverse, km_degradation, ki, uptake_rate, ): # Calculate kcat value to ensure sufficient supply to double fwd_fraction_basal = units.strip_empty_units( 1 / (1 + aa_conc_basal / ki) * np.prod(1 / (1 + kms / km_conc_basal)) ) rev_fraction_basal = units.strip_empty_units( 1 / ( 1 + km_reverse / aa_conc_basal * (1 + aa_conc_basal / km_degradation) ) ) deg_fraction_basal = units.strip_empty_units( 1 / ( 1 + km_degradation / aa_conc_basal * (1 + aa_conc_basal / km_reverse) ) ) loss_fraction_basal = rev_fraction_basal + deg_fraction_basal fwd_capacity_basal = fwd_enzymes_basal * fwd_fraction_basal rev_capacity_basal = rev_enzymes_basal * loss_fraction_basal fwd_fraction_with_aa = units.strip_empty_units( 1 / (1 + aa_conc_with_aa / ki) * np.prod(1 / (1 + kms / km_conc_with_aa)) ) rev_fraction_with_aa = units.strip_empty_units( 1 / ( 1 + km_reverse / aa_conc_with_aa * (1 + aa_conc_with_aa / km_degradation) ) ) deg_fraction_with_aa = units.strip_empty_units( 1 / ( 1 + km_degradation / aa_conc_with_aa * (1 + aa_conc_with_aa / km_reverse) ) ) loss_fraction_with_aa = rev_fraction_with_aa + deg_fraction_with_aa fwd_capacity_with_aa = fwd_enzymes_with_aa * fwd_fraction_with_aa rev_capacity_with_aa = rev_enzymes_with_aa * loss_fraction_with_aa supply_basal = basal_supply_mapping[amino_acid] supply_with_aa = with_aa_supply_mapping[amino_acid] downstream_basal = 0 downstream_with_aa = 0 for i, stoich in enumerate( self.aa_forward_stoich[self.aa_to_index[amino_acid], :] ): if stoich < 0: downstream_aa = aa_ids[i] downstream_basal += -stoich * fwd_rates[downstream_aa][0] downstream_with_aa += -stoich * fwd_rates[downstream_aa][1] for i, stoich in enumerate( self.aa_reverse_stoich[self.aa_to_index[amino_acid], :] ): if stoich < 0: downstream_aa = aa_ids[i] downstream_basal += stoich * rev_rates[downstream_aa][0] downstream_with_aa += stoich * rev_rates[downstream_aa][1] uptake = ( units.mmol / units.g / units.h * uptake_rate * cell_specs["with_aa"]["avgCellDryMassInit"] ).asNumber(units.count * K_CAT_UNITS) balance_basal = supply_basal + downstream_basal balance_with_aa = supply_with_aa + downstream_with_aa - uptake A = np.array( [ [fwd_capacity_basal, -rev_capacity_basal], [fwd_capacity_with_aa, -rev_capacity_with_aa], ] ) b = np.array([balance_basal, balance_with_aa]) try: kcat_fwd, kcat_rev = np.linalg.solve(A, b) except np.linalg.LinAlgError: if VERBOSE: print( f"Warning: could not solve directly for {amino_acid} kcats - switching to least squares" ) kcat_fwd, kcat_rev = np.linalg.lstsq(A, b, rcond=None)[0] fwd_rate = ( kcat_fwd * fwd_enzymes_basal * fwd_fraction_basal, kcat_fwd * fwd_enzymes_with_aa * fwd_fraction_with_aa, ) rev_rate = ( kcat_rev * rev_enzymes_basal * rev_fraction_basal, kcat_rev * rev_enzymes_with_aa * rev_fraction_with_aa, ) deg_rate = ( kcat_rev * rev_enzymes_basal * deg_fraction_basal, kcat_rev * rev_enzymes_with_aa * deg_fraction_with_aa, ) return kcat_fwd, kcat_rev, fwd_rate, rev_rate, deg_rate, uptake # Fit forward and reverse kcats by adjusting the uptake rate def objective(aa, uptake, kcat_fwd, kcat_rev): aa = aa[:-3] diffs = np.array( [ measured_uptake_rates[aa] - uptake, fwd_kcat_targets.get(aa, default_fwd_target) - kcat_fwd, 0 if units.isfinite(km_reverse) else kcat_rev, # no penalty if reverse, minimize if degradation ] ) weights = np.array([1000, 1, 1]) return np.linalg.norm(weights * diffs) # TODO (travis): use a more rigorous method to fit the kcats (eg. gradient descent) # It would be better to include all amino acids in an objective and solve for all # kcats iteratively instead of solving for kcats for a single amino acid and then the next best_objective = None kcat_fwd = None n_factors = 500 if VERBOSE: print("uptake:") for factor in np.logspace(-1, 1, n_factors): results = calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, kms, km_reverse, km_degradation, ki, factor * measured_uptake_rates[amino_acid[:-3]], ) new_kcat_fwd, new_kcat_rev, *_ = results if VERBOSE: print(f"\t{factor:.2f}:\t{new_kcat_fwd:5.1f}\t{new_kcat_rev:5.1f}") if new_kcat_fwd >= 0 and new_kcat_rev >= 0: new_objective = objective( amino_acid, factor * measured_uptake_rates[amino_acid[:-3]], new_kcat_fwd, new_kcat_rev, ) if best_objective is None or new_objective < best_objective: kcat_fwd, kcat_rev, fwd_rate, rev_rate, deg_rate, uptake = ( results ) data["kcat"] = kcat_fwd * K_CAT_UNITS best_objective = new_objective # Vary input parameters for kcat calculations for debugging purposes if VERBOSE: print("KMs:") for factor in np.logspace(-1, 1, n_factors): new_kcat_fwd, new_kcat_rev, *_ = calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, factor * kms, km_reverse, km_degradation, ki, measured_uptake_rates[amino_acid[:-3]], ) print(f"\t{factor:.2f}:\t{new_kcat_fwd:5.1f}\t{new_kcat_rev:5.1f}") print("km_reverse:") for factor in np.logspace(-1, 1, n_factors): new_kcat_fwd, new_kcat_rev, *_ = calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, kms, factor * km_reverse, km_degradation, ki, measured_uptake_rates[amino_acid[:-3]], ) print(f"\t{factor:.2f}:\t{new_kcat_fwd:5.1f}\t{new_kcat_rev:5.1f}") print("km_degradation:") for factor in np.logspace(-1, 1, n_factors): new_kcat_fwd, new_kcat_rev, *_ = calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, kms, km_reverse, factor * km_degradation, ki, measured_uptake_rates[amino_acid[:-3]], ) print(f"\t{factor:.2f}:\t{new_kcat_fwd:5.1f}\t{new_kcat_rev:5.1f}") print("ki:") for factor in np.logspace(-1, 1, n_factors): new_kcat_fwd, new_kcat_rev, *_ = calc_kcats( aa_conc_basal, km_conc_basal, aa_conc_with_aa, km_conc_with_aa, kms, km_reverse, km_degradation, factor * ki, measured_uptake_rates[amino_acid[:-3]], ) print(f"\t{factor:.2f}:\t{new_kcat_fwd:5.1f}\t{new_kcat_rev:5.1f}") print(f"*** {amino_acid}: {kcat_fwd:5.1f} {kcat_rev:5.1f} ***") if kcat_fwd is None: raise ValueError( "Could not find positive forward and reverse" f" kcat for {amino_acid}. Run with VERBOSE to check input" " parameters like KM and KI or check concentrations." ) aa_enzymes += fwd_enzymes + rev_enzymes enzyme_to_aa_fwd += [amino_acid] * len(fwd_enzymes) + [None] * len( rev_enzymes ) enzyme_to_aa_rev += [None] * len(fwd_enzymes) + [amino_acid] * len( rev_enzymes ) aa_kcats_fwd[amino_acid] = kcat_fwd aa_kcats_rev[amino_acid] = kcat_rev aa_kis[amino_acid] = ki.asNumber(METABOLITE_CONCENTRATION_UNITS) upstream_aas_for_km[amino_acid] = upstream_aa aa_upstream_kms[amino_acid] = kms.asNumber(METABOLITE_CONCENTRATION_UNITS) aa_reverse_kms[amino_acid] = km_reverse.asNumber( METABOLITE_CONCENTRATION_UNITS ) aa_degradation_kms[amino_acid] = km_degradation.asNumber( METABOLITE_CONCENTRATION_UNITS ) fwd_rates[amino_acid] = fwd_rate rev_rates[amino_acid] = rev_rate deg_rates[amino_acid] = deg_rate calculated_uptake_rates[amino_acid] = uptake self.aa_enzymes = np.unique(aa_enzymes) self.aa_kcats_fwd = np.array([aa_kcats_fwd[aa] for aa in aa_ids]) self.aa_kcats_rev = np.array([aa_kcats_rev[aa] for aa in aa_ids]) self.aa_kis = np.array([aa_kis[aa] for aa in aa_ids]) self.aa_reverse_kms = np.array([aa_reverse_kms[aa] for aa in aa_ids]) self.aa_degradation_kms = np.array([aa_degradation_kms[aa] for aa in aa_ids]) # Import inhibition of transporters rich_conc = np.array( [with_aa_conc[aa].asNumber(METABOLITE_CONCENTRATION_UNITS) for aa in aa_ids] ) self.aa_import_kis: np.ndarray = ( rich_conc.copy() ) # Assume this conc is the inhibition constant: TODO: find KIs saturation = 1 / (1 + rich_conc / self.aa_import_kis) self.specific_import_rates: np.ndarray = np.array( [calculated_uptake_rates[aa] for aa in aa_ids] ) / cell_specs["with_aa"]["avgCellDryMassInit"].asNumber(DRY_MASS_UNITS) self.max_specific_import_rates = self.specific_import_rates / saturation # KMs for upstream amino acids upstream_kms = [aa_upstream_kms[aa] for aa in aa_ids] upstream_aas = [upstream_aas_for_km[aa] for aa in aa_ids] self.aa_upstream_kms = np.zeros((n_aas, n_aas)) for i, (kms, aas) in enumerate(zip(upstream_kms, upstream_aas)): for km, aa in zip(kms, aas): self.aa_upstream_kms[i, self.aa_to_index[aa]] = km # Convert enzyme counts to an amino acid basis via dot product (counts @ self.enzyme_to_amino_acid) self.enzyme_to_amino_acid_fwd = np.zeros((len(self.aa_enzymes), n_aas)) self.enzyme_to_amino_acid_rev = np.zeros((len(self.aa_enzymes), n_aas)) enzyme_mapping = {e: i for i, e in enumerate(self.aa_enzymes)} aa_mapping = {a: i for i, a in enumerate(aa_ids)} for enzyme, fwd, rev in zip(aa_enzymes, enzyme_to_aa_fwd, enzyme_to_aa_rev): if fwd is not None: self.enzyme_to_amino_acid_fwd[ enzyme_mapping[enzyme], aa_mapping[fwd] ] = 1 if rev is not None: self.enzyme_to_amino_acid_rev[ enzyme_mapping[enzyme], aa_mapping[rev] ] = 1 # Concentrations for reference in analysis plot conversion = ( sim_data.constants.cell_density / sim_data.constants.n_avogadro * sim_data.mass.cell_dry_mass_fraction ) aa_enzymes_basal_idx = bulk_name_to_idx(self.aa_enzymes, basal_container["id"]) basal_counts = counts(basal_container, aa_enzymes_basal_idx) aa_enzymes_with_aa_idx = bulk_name_to_idx( self.aa_enzymes, with_aa_container["id"] ) with_aa_counts = counts(with_aa_container, aa_enzymes_with_aa_idx) self.aa_supply_enzyme_conc_with_aa = ( conversion * with_aa_counts / cell_specs["with_aa"]["avgCellDryMassInit"] ) self.aa_supply_enzyme_conc_basal = ( conversion * basal_counts / cell_specs["basal"]["avgCellDryMassInit"] ) # Check calculations that could end up negative neg_idx = np.where(self.max_specific_import_rates < 0)[0] if len(neg_idx): aas = ", ".join([aa_ids[idx] for idx in neg_idx]) print(f"{self.max_specific_import_rates = }") raise ValueError( f"Import rate was determined to be negative for {aas}." " Check input parameters like supply and synthesis or enzyme expression." )
[docs] def get_pathway_enzyme_counts_per_aa( self, enzyme_counts: np.ndarray ) -> Tuple[np.ndarray, np.ndarray]: """ Get the counts of enzymes for forward and reverse reactions in the amino acid synthesis network based on all of the enzymes used in the network. Useful to get the counts to pass to amino_acid_synthesis() from counts based on self.aa_enzymes. Args: enzyme_counts: counts of all enzymes in the amino acid network Returns: 2-element tuple containing - counts_per_aa_fwd: counts of enzymes for the forward reaction for each amino acid - counts_per_aa_rev: counts of enzymes for the reverse reaction for each amino acid """ counts_per_aa_fwd = enzyme_counts @ self.enzyme_to_amino_acid_fwd counts_per_aa_rev = enzyme_counts @ self.enzyme_to_amino_acid_rev return counts_per_aa_fwd, counts_per_aa_rev
[docs] def amino_acid_synthesis( self, counts_per_aa_fwd: npt.NDArray[np.int64], counts_per_aa_rev: npt.NDArray[np.int64], aa_conc: Union[units.Unum, npt.NDArray[np.float64]], ) -> tuple[ npt.NDArray[np.float64], npt.NDArray[np.float64], npt.NDArray[np.float64] ]: """ Calculate the net rate of synthesis for amino acid pathways (can be negative with reverse reactions). Args: counts_per_aa_fwd: counts for enzymes in forward reactions for each amino acid counts_per_aa_rev: counts for enzymes in loss reactions for each amino acid aa_conc: concentrations of each amino acid with mol/volume units Returns: 3-element tuple containing - synthesis: net rate of synthesis for each amino acid pathway. array is unitless but represents counts of amino acid per second - forward_fraction: saturated fraction for forward reactions - loss_fraction: saturated fraction for loss reactions .. note:: Currently does not match saturation terms used in calc_kcats since it assumes only a reverse or degradation KM exists for simpler calculations """ # Convert to appropriate arrays if units.hasUnit(aa_conc): aa_conc = aa_conc.asNumber(METABOLITE_CONCENTRATION_UNITS) return amino_acid_synthesis_jit( counts_per_aa_fwd, counts_per_aa_rev, aa_conc, self.aa_upstream_kms, self.aa_kis, self.aa_reverse_kms, self.aa_degradation_kms, self.aa_forward_stoich, self.aa_kcats_fwd, self.aa_reverse_stoich, self.aa_kcats_rev, )
[docs] def amino_acid_export( self, aa_transporters_counts: npt.NDArray[np.int64], aa_conc: Union[units.Unum, npt.NDArray[np.float64]], mechanistic_uptake: bool, ) -> npt.NDArray[np.float64]: """ Calculate the rate of amino acid export. Args: aa_transporters_counts: counts of each transporter aa_conc: concentrations of each amino acid with mol/volume units mechanistic_uptake: if true, the uptake is calculated based on transporters Returns: Rate of export for each amino acid (unitless but represents counts of amino acid per second) """ if units.hasUnit(aa_conc): aa_conc = aa_conc.asNumber(METABOLITE_CONCENTRATION_UNITS) return amino_acid_export_jit( aa_transporters_counts, aa_conc, mechanistic_uptake, self.aa_to_exporters_matrix, self.export_kcats_per_aa, self.aa_export_kms, )
[docs] def amino_acid_import( self, aa_in_media: npt.NDArray[np.bool_], dry_mass: units.Unum, internal_aa_conc: Union[units.Unum, npt.NDArray[np.float64]], aa_transporters_counts: npt.NDArray[np.int64], mechanisitic_uptake: bool, ) -> npt.NDArray[np.float64]: """ Calculate the rate of amino acid uptake. Args: aa_in_media: bool for each amino acid being present in current media dry_mass: current dry mass of the cell, with mass units internal_aa_conc: internal concentrations of amino acids aa_transporters_counts: counts of each transporter mechanisitic_uptake: if true, the uptake is calculated based on transporters Returns: Rate of uptake for each amino acid (unitless but represents counts of amino acid per second) """ if units.hasUnit(internal_aa_conc): internal_aa_conc = internal_aa_conc.asNumber(METABOLITE_CONCENTRATION_UNITS) dry_mass = dry_mass.asNumber(DRY_MASS_UNITS) return amino_acid_import_jit( aa_in_media, dry_mass, internal_aa_conc, aa_transporters_counts, mechanisitic_uptake, self.aa_import_kis, self.aa_to_importers_matrix, self.import_kcats_per_aa, self.max_specific_import_rates, )
[docs] def get_amino_acid_conc_conversion(self, conc_units): return units.strip_empty_units(conc_units / METABOLITE_CONCENTRATION_UNITS)
[docs] @staticmethod def extract_reactions( raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli" ) -> tuple[ list[str], dict[str, dict[str, int]], list[str], dict[str, list[str]], dict[str, str], ]: """ Extracts reaction data from raw_data to build metabolism reaction network with stoichiometry, reversibility and enzyme catalysts. Args: raw_data: knowledge base data sim_data: simulation data Returns: 5-element tuple containing - base_rxn_ids: list of base reaction IDs from which reaction IDs were derived from - reaction_stoich: stoichiometry of metabolites for each reaction:: {reaction ID: {metabolite ID with location tag: stoichiometry}} - reversible_reactions: reaction IDs for reactions that have a reverse complement, does not have reverse tag - reaction_catalysts: enzyme catalysts for each reaction with known catalysts, likely a subset of reactions in stoich:: {reaction ID: enzyme IDs with location tag} - rxn_id_to_base_rxn_id: mapping from reaction IDs to the IDs of the base reactions they were derived from:: {reaction ID: base ID} """ compartment_ids_to_abbreviations = { comp["id"]: comp["abbrev"] for comp in raw_data.compartments } compartment_ids_to_abbreviations.update(UNDEFINED_COMPARTMENT_IDS_TO_ABBREVS) valid_directions = {"L2R", "R2L", "BOTH"} forward_directions = {"L2R", "BOTH"} reverse_directions = {"R2L", "BOTH"} metabolite_ids = {met["id"] for met in cast(Any, raw_data).metabolites} # Build mapping from each complexation subunit to all downstream # complexes containing the subunit, including itself # Start by building mappings from subunits to complexes that are # directly formed from the subunit through a single reaction subunit_id_to_parent_complexes: dict[str, list[str]] = {} for comp_reaction in itertools.chain( cast(Any, raw_data).complexation_reactions, cast(Any, raw_data).equilibrium_reactions, ): complex_id = None # Find ID of complex for mol_id, coeff in comp_reaction["stoichiometry"].items(): if coeff > 0: complex_id = mol_id break assert complex_id is not None # Map each subunit to found complex for mol_id, coeff in comp_reaction["stoichiometry"].items(): if mol_id == complex_id or mol_id in metabolite_ids: continue elif mol_id in subunit_id_to_parent_complexes: subunit_id_to_parent_complexes[mol_id].append(complex_id) else: subunit_id_to_parent_complexes[mol_id] = [complex_id] # Recursive function that returns a list of all downstream complexes # containing the given subunit, including itself def get_all_complexes(subunit_id): all_downstream_complex_ids = [subunit_id] if subunit_id not in subunit_id_to_parent_complexes: return all_downstream_complex_ids # Get downstream complexes of all parent complexes for parent_complex_id in subunit_id_to_parent_complexes[subunit_id]: all_downstream_complex_ids.extend(get_all_complexes(parent_complex_id)) # Remove duplicates return sorted(set(all_downstream_complex_ids)) subunit_id_to_all_downstream_complexes = { subunit_id: get_all_complexes(subunit_id) for subunit_id in subunit_id_to_parent_complexes.keys() } # Initialize variables to store reaction information all_base_rxns = set() reaction_stoich = {} reversible_reactions = [] reaction_catalysts = {} rxn_id_to_base_rxn_id = {} # Load and parse reaction information from raw_data for reaction in cast(Any, raw_data).metabolic_reactions: reaction_id = reaction["id"] stoich = reaction["stoichiometry"] direction = reaction["direction"] if len(stoich) <= 1: raise Exception( "Invalid biochemical reaction: {}, {}".format(reaction_id, stoich) ) if direction not in valid_directions: raise InvalidReactionDirectionError( f"The direction {direction} given for reaction {reaction_id} is invalid." ) forward = direction in forward_directions reverse = direction in reverse_directions def convert_compartment_tags(met_id): new_met_id = met_id for comp_id, comp_abbrev in compartment_ids_to_abbreviations.items(): new_met_id = new_met_id.replace(f"[{comp_id}]", f"[{comp_abbrev}]") return new_met_id # All protein complexes that contain an enzyme subunit are assumed # to retain the enzyme's catalytic activity catalysts_for_this_rxn = [] all_potential_catalysts = [] for catalyst in reaction["catalyzed_by"]: all_potential_catalysts.extend( subunit_id_to_all_downstream_complexes.get(catalyst, [catalyst]) ) for catalyst in sorted(set(all_potential_catalysts)): if sim_data.getter.is_valid_molecule(catalyst): catalysts_with_loc = catalyst + sim_data.getter.get_compartment_tag( catalyst ) catalysts_for_this_rxn.append(catalysts_with_loc) # If we don't have the catalyst in our reconstruction, drop it else: if VERBOSE: print( "Skipping catalyst {} for {} since it is not in the model".format( catalyst, reaction_id ) ) # Get base reaction ID of this reaction # If reaction ID does not end with a dot, the given reaction ID is # already a base reaction ID if reaction_id[-1] != ".": base_reaction_id = reaction_id # If reaction ID ends with a dot, find the base reaction ID based # on the following rules (provided by EcoCyc): # The parsing instructions for obtaining the base rxn-ID are, effectively: # 1: Check whether the full rxn-ID ends with a dot. # 2: If there is no dot at the end, it is already a base rxn-ID. # Otherwise, find the position of a second dot, to the left of the dot at the end. # 3: Extract the string between the last dot and the second to last dot. # If this intervening string consists of only digits, then convert this string to an integer. # In this case, this rxn-ID stands for a generic rxn, and has extra suffixes # that need to be trimmed off, to retrieve the base rxn-ID. # The extracted integer indicates the length of the suffixes to be trimmed off. # 4: To find the end position of the base rxn-ID, subtract the integer (obtained by 3: ) # from the position of the second dot (obtained by 2: ) . # 5: Retrieve the base rxn-ID, which is the substring from the very left (position 0) # to the end position (obtained by 4: ) . else: reaction_id_split = reaction_id[:-1].split(".") suffix_length = int(reaction_id_split[-1]) base_reaction_id = ".".join(reaction_id_split[:-1])[:-suffix_length] if forward: reaction_stoich[reaction_id] = { convert_compartment_tags(moleculeID): stoichCoeff for moleculeID, stoichCoeff in stoich.items() } if len(catalysts_for_this_rxn) > 0: reaction_catalysts[reaction_id] = catalysts_for_this_rxn rxn_id_to_base_rxn_id[reaction_id] = base_reaction_id if reverse: reverse_reaction_id = REVERSE_REACTION_ID.format(reaction_id) reaction_stoich[reverse_reaction_id] = { convert_compartment_tags(moleculeID): -stoichCoeff for moleculeID, stoichCoeff in stoich.items() } if len(catalysts_for_this_rxn) > 0: reaction_catalysts[reverse_reaction_id] = list( catalysts_for_this_rxn ) rxn_id_to_base_rxn_id[reverse_reaction_id] = base_reaction_id if forward and reverse: reversible_reactions.append(reaction_id) if base_reaction_id not in all_base_rxns: all_base_rxns.add(base_reaction_id) base_rxn_ids = sorted(list(all_base_rxns)) return ( base_rxn_ids, reaction_stoich, reversible_reactions, reaction_catalysts, rxn_id_to_base_rxn_id, )
[docs] @staticmethod def match_reaction( stoich: dict[str, dict[str, int]], catalysts: dict[str, list[str]], rxn_to_match: str, enz: str, mets: list[str], direction: Optional[str] = None, ) -> list[str]: """ Matches a given reaction (rxn_to_match) to reactions that exist in stoich given that enz is known to catalyze the reaction and mets are reactants in the reaction. Can perform a fuzzy reaction match since rxn_to_match just needs to be part of the actual reaction name to match specific instances of a reaction. (eg. rxn_to_match="ALCOHOL-DEHYDROG-GENERIC-RXN" can match "ALCOHOL-DEHYDROG-GENERIC-RXN-ETOH/NAD//ACETALD/NADH/PROTON.30."). Args: stoich: stoichiometry of metabolites for each reaction:: {reaction ID: {metabolite ID with location tag: stoichiometry}} catalysts: enzyme catalysts for each reaction with known catalysts, likely a subset of reactions in stoich:: {reaction ID: enzyme IDs with location tag} rxn_to_match: reaction ID from kinetics to match to existing reactions enz: enzyme ID with location tag mets: metabolite IDs with no location tag from kinetics direction: reaction directionality, ``'forward'`` or ``'reverse'`` or ``None`` Returns: Matched reaction IDs in stoich """ # Mapping to handle instances of metabolite classes in kinetics # Keys: specific molecules in kinetics file # Values: class of molecules in reactions file that contain the key class_mets = { "RED-THIOREDOXIN-MONOMER": "Red-Thioredoxin", "RED-THIOREDOXIN2-MONOMER": "Red-Thioredoxin", "RED-GLUTAREDOXIN": "Red-Glutaredoxins", "GRXB-MONOMER": "Red-Glutaredoxins", "GRXC-MONOMER": "Red-Glutaredoxins", "OX-FLAVODOXIN1": "Oxidized-flavodoxins", "OX-FLAVODOXIN2": "Oxidized-flavodoxins", } # Match full reaction name from partial reaction in kinetics. Must # also match metabolites since there can be multiple reaction instances. match = False match_candidates = [] if rxn_to_match in stoich: match_candidates.append(rxn_to_match) else: for long_rxn, long_mets in stoich.items(): if rxn_to_match in long_rxn and not long_rxn.endswith(REVERSE_TAG): match = True stripped_enzs = {e[:-3] for e in catalysts.get(long_rxn, [])} stripped_mets = {m[:-3] for m in long_mets} if ( np.all([class_mets.get(m, m) in stripped_mets for m in mets]) and enz in stripped_enzs ): match_candidates.append(long_rxn) if len(match_candidates) == 0: if VERBOSE: if match: print( "Partial reaction match: {} {} {} {} {}".format( rxn_to_match, enz, stripped_enzs, mets, stripped_mets ) ) else: print("No reaction match: {}".format(rxn_to_match)) # Determine direction of kinetic reaction from annotation or # metabolite stoichiometry. rxn_matches = [] for rxn in match_candidates: reverse_rxn = REVERSE_REACTION_ID.format(rxn) reverse_rxn_exists = reverse_rxn in stoich if direction: reverse = direction == "reverse" else: s = {k[:-3]: v for k, v in stoich.get(rxn, {}).items()} direction_ = np.unique( np.sign([s.get(class_mets.get(m, m), 0) for m in mets]) ) if len(direction_) == 0 and not reverse_rxn_exists: reverse = False elif len(direction_) != 1 or direction_[0] == 0: if VERBOSE: print( "Conflicting directionality: {} {} {}".format( rxn, mets, direction_ ) ) continue else: reverse = direction_[0] > 0 # Verify a reverse reaction exists in the model if reverse: if reverse_rxn_exists: rxn_matches.append(reverse_rxn) continue else: if VERBOSE: print("No reverse reaction: {} {}".format(rxn, mets)) continue rxn_matches.append(rxn) return sorted(rxn_matches)
[docs] @staticmethod def temperature_adjusted_kcat( kcat: Unum, temp: Union[float, str] = "" ) -> npt.NDArray[np.float64]: """ Args: kcat: enzyme turnover number(s) (1 / time) temp: temperature of measurement, defaults to 25 if '' Returns: Temperature adjusted kcat values, in units of 1/s """ if isinstance(temp, str): temp = 25 return 2 ** ((37.0 - temp) / 10.0) * kcat.asNumber(K_CAT_UNITS)
[docs] @staticmethod def _construct_default_saturation_equation( mets: list[str], kms: list[float], kis: list[float], known_mets: Iterable[str] ) -> str: """ Args: mets: metabolite IDs with location tag for KM and KI parameters ordered to match order of kms then kis kms: KM parameters associated with mets kis: KI parameters associated with mets known_mets: metabolite IDs with location tag with known concentrations Returns: Saturation equation with metabolites to replace delimited by double quote (e.g. "metabolite") """ # Check input dimensions n_params = len(kms) + len(kis) if n_params == 0: return "1" if n_params != len(mets): if VERBOSE: print("Saturation parameter mismatch: {} {} {}".format(mets, kms, kis)) return "1" terms = [] # Add KM terms for m, k in zip(mets, kms): if m in known_mets: terms.append('1+{}/"{}"'.format(k, m)) elif VERBOSE: print("Do not have concentration for {} with KM={}".format(m, k)) # Add KI terms for m, k in zip(mets[len(kms) :], kis): if m in known_mets: terms.append('1+"{}"/{}'.format(m, k)) elif VERBOSE: print("Do not have concentration for {} with KI={}".format(m, k)) # Enclose groupings if being multiplied together if len(terms) > 1: terms[0] = "(" + terms[0] terms[-1] += ")" elif len(terms) == 0: return "1" return "1/({})".format(")*(".join(terms))
[docs] @staticmethod def _extract_custom_constraint( constraint: dict[str, Any], reactant_tags: dict[str, str], product_tags: dict[str, str], known_mets: set[str], ) -> tuple[Optional[npt.NDArray[np.float64]], list[str]]: """ Args: constraint: values defining a kinetic constraint:: {'customRateEquation' (str): mathematical representation of rate (must contain 'kcat*E'), 'customParameterVariables' (dict[str, str]): mapping of variable names in the rate equation to metabolite IDs without location tags (must contain 'E'), 'customParameterConstants' (list[str]): constant strings in the rate equation that correspond to values (must contain 'kcat'), 'customParameterConstantValues' (list[float]): values for each of the constant strings, 'Temp' (float or ''): temperature of measurement} reactant_tags: mapping of molecule IDs without a location tag to molecule IDs with a location tag for all reactants product_tags: mapping of molecule IDs without a location tag to molecule IDs with a location tag for all products known_mets: molecule IDs with a location tag for molecules with known concentrations Returns: 2-element tuple containing - kcats: temperature adjusted kcat value, in units of 1/s - saturation: saturation equation with metabolites to replace delimited by double quote (eg. "metabolite") """ equation = constraint["customRateEquation"] variables = constraint["customParameterVariables"] constant_keys = constraint["customParameterConstants"] constant_values = constraint["customParameterConstantValues"] temp = constraint["Temp"] # Need to have these in the constraint kcat_str = "kcat" enzyme_str = "E" capacity_str = "{}*{}".format(kcat_str, enzyme_str) # Need to replace these symbols in equations symbol_sub = { "^": "**", } # Make sure kcat exists if kcat_str not in constant_keys: if VERBOSE: print( "Missing {} in custom constants: {}".format(kcat_str, constant_keys) ) return None, [] custom_kcat = ( 1 / units.s * np.array([constant_values[constant_keys.index(kcat_str)]]) ) kcats = Metabolism.temperature_adjusted_kcat(custom_kcat, temp) # Make sure equation can be parsed, otherwise just return kcat if enzyme_str not in variables: if VERBOSE: print( "Missing enzyme key ({}) in custom variables: {}".format( enzyme_str, variables ) ) return kcats, [] if capacity_str not in equation: if VERBOSE: print( "Expected to find {} in custom equation: {}".format( capacity_str, equation ) ) return kcats, [] if len(constant_keys) != len(constant_values): if VERBOSE: print( "Mismatch between constants: {} {}".format( constant_keys, constant_values ) ) return kcats, [] variables_with_tags = { k: reactant_tags.get(v, product_tags.get(v, None)) for k, v in variables.items() if k != enzyme_str and (v in reactant_tags or v in product_tags) } # Substitute values into custom equations ## Replace terms with known constant values or sim molecule IDs with concentrations custom_subs = {k: str(v) for k, v in zip(constant_keys, constant_values)} custom_subs.update( { k: '"{}"'.format(v) for k, v in variables_with_tags.items() if v in known_mets } ) ## Remove capacity to get only saturation term new_equation = equation.replace(capacity_str, "1") ## Tokenize equation to terms and symbols parsed_variables = re.findall(r"\w*", new_equation)[ :-1 ] # Remove trailing empty match ## Ensure valid input of known variables or a float term for v in parsed_variables: if not (v == "" or v in custom_subs): try: float(v) except ValueError: if VERBOSE: print( "Unknown value encountered in custom equation {}: {}".format( equation, v ) ) return kcats, [] parsed_symbols = re.findall(r"\W", new_equation) tokenized_equation = np.array(parsed_variables) symbol_idx_mask = tokenized_equation == "" ## Verify tokenized equation matches original before replacements tokenized_equation[symbol_idx_mask] = parsed_symbols if "".join(tokenized_equation) != new_equation: if VERBOSE: print("Error parsing custom equation: {}".format(equation)) return kcats, [] ## Perform replacement of symbols tokenized_equation[symbol_idx_mask] = [ symbol_sub.get(s, s) for s in parsed_symbols ] # Reconstruct saturation equation with replacements saturation = [ "".join([custom_subs.get(token, token) for token in tokenized_equation]) ] return kcats, saturation
[docs] @staticmethod def extract_kinetic_constraints( raw_data: KnowledgeBaseEcoli, sim_data: "SimulationDataEcoli", stoich: Optional[dict[str, dict[str, int]]] = None, catalysts: Optional[dict[str, list[str]]] = None, known_metabolites: Optional[Set[str]] = None, ) -> dict[tuple[str, str], dict[str, list[Any]]]: """ Load and parse kinetic constraint information from raw_data Args: raw_data: knowledge base data sim_data: simulation data stoich: stoichiometry of metabolites for each reaction (if ``None``, data is loaded from ``raw_data`` and ``sim_data``):: {reaction ID: {metabolite ID with location tag: stoichiometry}} catalysts: enzyme catalysts for each reaction with known catalysts, likely a subset of reactions in ``stoich`` (if ``None``, data is loaded from ``raw_data`` and ``sim_data``:: {reaction ID: enzyme IDs with location tag} known_metabolites: metabolites with known concentrations Returns: Valid kinetic constraints for each reaction/enzyme pair:: {(reaction ID, enzyme with location tag): { 'kcat': kcat values (list[float]), 'saturation': saturation equations (list[str]) }} """ # Load data for optional args if needed if stoich is None or catalysts is None: _, loaded_stoich, _, loaded_catalysts, _ = Metabolism.extract_reactions( raw_data, sim_data ) if stoich is None: stoich = loaded_stoich if catalysts is None: catalysts = loaded_catalysts known_metabolites_ = set() if known_metabolites is None else known_metabolites constraints: dict[tuple[str, str], dict[str, list]] = {} for constraint in cast(Any, raw_data).metabolism_kinetics: rxn = constraint["reactionID"] enzyme = constraint["enzymeID"] metabolites = constraint["substrateIDs"] direction = constraint["direction"] kms = list(constraint["kM"].asNumber(KINETIC_CONSTRAINT_CONC_UNITS)) kis = list(constraint["kI"].asNumber(KINETIC_CONSTRAINT_CONC_UNITS)) n_reactants = len(metabolites) - len(kis) matched_rxns = Metabolism.match_reaction( stoich, catalysts, rxn, enzyme, metabolites[:n_reactants], direction ) for matched_rxn in matched_rxns: # Ensure enzyme catalyzes reaction in model enzymes_tag_conversion = { e[:-3]: e for e in catalysts.get(matched_rxn, []) } if enzyme not in enzymes_tag_conversion: if VERBOSE: print("{} does not catalyze {}".format(enzyme, matched_rxn)) continue # Update metabolites with a location tag from the reaction # First look in reactants but some products can inhibit reactant_tags = { k[:-3]: k for k, v in stoich[matched_rxn].items() if v < 0 } product_tags = { k[:-3]: k for k, v in stoich[matched_rxn].items() if v > 0 } mets_with_tag = [ reactant_tags.get(met, product_tags.get(met, "")) for met in metabolites if met in reactant_tags or met in product_tags ] if len(mets_with_tag) != len(metabolites): # Warn if verbose but no continue since we can still use kcat if VERBOSE: print( "Could not match all metabolites: {} {}".format( metabolites, mets_with_tag ) ) # Extract kcat and saturation parameters if constraint["rateEquationType"] == "custom": kcats, saturation = Metabolism._extract_custom_constraint( constraint, reactant_tags, product_tags, known_metabolites_ ) if kcats is None: continue else: kcats = Metabolism.temperature_adjusted_kcat( constraint["kcat"], constraint["Temp"] ) if len(kcats) > 1: if len(kcats) != len(kms) or len(kms) != len(mets_with_tag): if VERBOSE: print( "Could not align kcats and kms: {} {} {} {}".format( rxn, kcats, kms, mets_with_tag ) ) continue saturation = [ Metabolism._construct_default_saturation_equation( [m], [km], [], known_metabolites_ ) for m, km in zip(mets_with_tag, kms) ] else: saturation = [ Metabolism._construct_default_saturation_equation( mets_with_tag, kms, kis, known_metabolites_ ) ] saturation = [s for s in saturation if s != "1"] # Add new kcats and saturation terms for the enzymatic reaction key = (matched_rxn, enzymes_tag_conversion[enzyme]) entries = constraints.get(key, {}) entries["kcat"] = entries.get("kcat", []) + list(kcats) entries["saturation"] = entries.get("saturation", []) + saturation constraints[key] = entries return constraints
[docs] @staticmethod def _replace_enzyme_reactions( constraints: dict[tuple[str, str], dict[str, list[Any]]], stoich: dict[str, dict[str, int]], rxn_catalysts: dict[str, list[str]], reversible_rxns: list[str], rxn_id_to_compiled_id: dict[str, str], ) -> tuple[ dict[str, Any], dict[str, dict[str, int]], dict[str, list[str]], list[str], dict[str, str], ]: """ Modifies reaction IDs in data structures to duplicate reactions with kinetic constraints and multiple enzymes. Args: constraints: valid kinetic constraints for each reaction/enzyme pair:: {(reaction ID, enzyme with location tag): { 'kcat': kcat values (list[float]), 'saturation': saturation equations (list[str]) }} stoich: stoichiometry of metabolites for each reaction (if None, data is loaded from raw_data and sim_data):: {reaction ID: {metabolite ID with location tag: stoichiometry}} rxn_catalysts: enzyme catalysts for each reaction with known catalysts, likely a subset of reactions in stoich (if None, data is loaded from raw_data and sim_data):: {reaction ID: enzyme IDs with location tag} reversible_rxns: reaction IDs for reactions that have a reverse complement, does not have reverse tag rxn_id_to_compiled_id: mapping from reaction IDs to the IDs of the original reactions they were derived from Returns: 5-element tuple containing - new_constraints: valid kinetic constraints for each reaction:: {reaction ID: { 'enzyme': enzyme catalyst (str), 'kcat': kcat values (list[float]), 'saturation': saturation equations (list[str]) }} - stoich: stoichiometry of metabolites for each reaction with updated reactions for enzyme catalyzed kinetic reactions:: {reaction ID: {metabolite ID with location tag: stoichiometry}} - rxn_catalysts: enzyme catalysts for each reaction with known catalysts, likely a subset of reactions in stoich with updated reactions for enzyme catalyzed kinetic reactions:: {reaction ID: enzyme IDs with location tag} - reversible_rxns: reaction IDs for reactions that have a reverse complement with updated reactions for enzyme catalyzed kinetic reactions, does not have reverse tag - rxn_id_to_compiled_id: mapping from reaction IDs to the IDs of the original reactions they were derived from, with updated reactions for enzyme catalyzed kinetic reactions """ new_constraints = {} n_catalysts = {rxn: len(catalysts) for rxn, catalysts in rxn_catalysts.items()} # Split out reactions that are kinetically constrained and that have # more than one enzyme that catalyzes the reaction for (rxn, enzyme), constraint in constraints.items(): if n_catalysts[rxn] > 1: # Create new reaction name with enzyme appended to the end if rxn.endswith(REVERSE_TAG): new_rxn = REVERSE_REACTION_ID.format( ENZYME_REACTION_ID.format(rxn[: -len(REVERSE_TAG)], enzyme[:-3]) ) else: new_rxn = ENZYME_REACTION_ID.format(rxn, enzyme[:-3]) # Add the new reaction to appropriate lists and dicts stoich[new_rxn] = copy(stoich[rxn]) rxn_catalysts[new_rxn] = [enzyme] if rxn in reversible_rxns: reversible_rxns.append(new_rxn) # Remove enzyme from old reaction and remove old reaction if no # more enzyme catalysts rxn_catalysts[rxn].pop(rxn_catalysts[rxn].index(enzyme)) if len(rxn_catalysts[rxn]) == 0: stoich.pop(rxn) rxn_catalysts.pop(rxn) if rxn in reversible_rxns: reversible_rxns.pop(reversible_rxns.index(rxn)) else: new_rxn = rxn rxn_id_to_compiled_id[new_rxn] = rxn_id_to_compiled_id[rxn] # noinspection PyTypeChecker new_constraints[new_rxn] = dict(constraints[(rxn, enzyme)], enzyme=enzyme) return ( new_constraints, stoich, rxn_catalysts, reversible_rxns, rxn_id_to_compiled_id, )
[docs] @staticmethod def _lambdify_constraints( constraints: dict[str, Any], ) -> tuple[ list[str], list[str], list[str], npt.NDArray[np.float64], str, str, npt.NDArray[np.bool_], ]: """ Creates str representations of kinetic terms to be used to create kinetic constraints that are returned with getKineticConstraints(). Args: constraints: valid kinetic constraints for each reaction:: {reaction ID: { 'enzyme': enzyme catalyst (str), 'kcat': kcat values (list[float]), 'saturation': saturation equations (list[str]) }} Returns: 7-element tuple containing - rxns: sorted reaction IDs for reactions with a kinetic constraint - enzymes: sorted enzyme IDs for enzymes that catalyze a kinetic reaction - substrates: sorted substrate IDs for substrates that are needed for kinetic saturation terms - all_kcats: (n rxns, 3) min, mean and max kcat value for each reaction - all_saturations: sympy str representation of a list of saturation terms (eg. '[s[0] / (1 + s[0]), 2 / (2 + s[1])]') - all_enzymes: sympy str representation of enzymes for each reaction (e.g. '[e[0], e[2], e[1]]') - constraint_is_kcat_only: True if reaction only has kcat values and no saturation terms """ # Ordered lists of constraint related IDs rxns = sorted(constraints) enzymes = sorted({c["enzyme"] for c in constraints.values()}) substrates = sorted( { match.strip('"') for c in constraints.values() for s in c["saturation"] for match in re.findall('".+?"', s) } ) # Mapping to replace molecule IDs with generic list strings enzyme_sub = {e: "e[{}]".format(i) for i, e in enumerate(enzymes)} substrate_sub = { '"{}"'.format(s): "s[{}]".format(i) for i, s in enumerate(substrates) } # Mapping to replace generic list strings with sympy variables. # Need separate mapping from above because sympy handles '[]' as indexing # so location tags are not parsed properly. enzyme_symbols = { "e": [sp.symbols("e[{}]".format(i)) for i in range(len(enzymes))] } substrate_symbols = { "s": [sp.symbols("s[{}]".format(i)) for i in range(len(substrates))] } # Values to return all_kcats = np.zeros((len(rxns), 3)) all_saturations = [] all_enzymes = [] constraint_is_kcat_only = [] # Extract out data from each constraint for i, rxn in enumerate(rxns): kcats = constraints[rxn]["kcat"] saturation = constraints[rxn]["saturation"] enzyme = constraints[rxn]["enzyme"] # Parse saturation equations into sympy format # If no saturation data is known, assume open range from 0 to 1 saturations = [] for sat in saturation: if sat == "1": continue for token, replace in substrate_sub.items(): sat = sat.replace(token, replace) saturations.append(parse_expr(sat, local_dict=substrate_symbols)) if len(saturations) == 0: saturations = [0, 1] constraint_is_kcat_only.append(True) else: constraint_is_kcat_only.append(False) # Save values for this constraint all_kcats[i, :] = [np.min(kcats), np.mean(kcats), np.max(kcats)] all_saturations.append(saturations) all_enzymes.append( parse_expr(enzyme_sub[enzyme], local_dict=enzyme_symbols) ) # Convert to str to save as class attr to be executed all_saturations = str(all_saturations) all_enzymes = str(all_enzymes) constraint_is_kcat_only = np.array(constraint_is_kcat_only) return ( rxns, enzymes, substrates, all_kcats, all_saturations, all_enzymes, constraint_is_kcat_only, )
[docs] def _is_transport_rxn(self, stoich: dict[str, int]) -> bool: """ Determines if the metabolic reaction with a given stoichiometry is a transport reactions that transports metabolites between different compartments. A metabolic reaction is considered to be a transport reaction if the substrate set and the product share the same metabolite tagged into different compartments. Args: stoich: Stoichiometry of the metabolic reaction:: { metabolite ID (str): stoichiometric coefficient (int) } Returns: True if given stoichiometry is a transport reaction """ is_transport_rxn = False # Get IDs of all substrates and products substrates, products = [], [] for mol_id, coeff in stoich.items(): if coeff < 0: substrates.append(mol_id) else: products.append(mol_id) # Get mapping from IDs to IDs without compartments substrates_tagged_to_no_tag = { mol_id: re.sub(r"\[.*]", "", mol_id) for mol_id in substrates } products_tagged_to_no_tag = { mol_id: re.sub(r"\[.*]", "", mol_id) for mol_id in products } overlap_no_tag = set(substrates_tagged_to_no_tag.values()) & set( products_tagged_to_no_tag.values() ) for mol_id_no_tag in list(overlap_no_tag): substrates_tagged = [ mol_tagged for mol_tagged in substrates if substrates_tagged_to_no_tag[mol_tagged] == mol_id_no_tag ] products_tagged = [ mol_tagged for mol_tagged in products if products_tagged_to_no_tag[mol_tagged] == mol_id_no_tag ] overlap_tagged = set(substrates_tagged) & set(products_tagged) # Tag reaction as a transport reaction if there is no overlap # between those substrates and products with locations included if len(overlap_tagged) == 0: is_transport_rxn = True break return is_transport_rxn
[docs] @njit def np_apply_along_axis(func1d, axis, arr): if arr.ndim != 2: raise RuntimeError("Array must have 2 dimensions.") if axis not in [0, 1]: raise RuntimeError("Axis must be 0 or 1.") if axis == 0: result = np.empty(arr.shape[1]) for i in range(len(result)): result[i] = func1d(arr[:, i]) else: result = np.empty(arr.shape[0]) for i in range(len(result)): result[i] = func1d(arr[i, :]) return result
[docs] @njit def np_prod(array, axis): return np_apply_along_axis(np.prod, axis, array)
[docs] @njit def amino_acid_synthesis_jit( counts_per_aa_fwd, counts_per_aa_rev, aa_conc, aa_upstream_kms, aa_kis, aa_reverse_kms, aa_degradation_kms, aa_forward_stoich, aa_kcats_fwd, aa_reverse_stoich, aa_kcats_rev, ): km_saturation = np_prod(1 / (1 + aa_upstream_kms / aa_conc), axis=1) # Determine saturation fraction for reactions forward_fraction = 1 / (1 + aa_conc / aa_kis) * km_saturation reverse_fraction = 1 / (1 + aa_reverse_kms / aa_conc) deg_fraction = 1 / (1 + aa_degradation_kms / aa_conc) loss_fraction = reverse_fraction + deg_fraction # Calculate synthesis rate synthesis = ( aa_forward_stoich.astype(np.float64)
[docs] @ (aa_kcats_fwd * counts_per_aa_fwd * forward_fraction).astype(np.float64) - aa_reverse_stoich.astype(np.float64) @ (aa_kcats_rev * counts_per_aa_rev * reverse_fraction).astype(np.float64) - aa_kcats_rev * counts_per_aa_rev * deg_fraction ) return synthesis, forward_fraction, loss_fraction
@njit def amino_acid_export_jit( aa_transporters_counts, aa_conc, mechanistic_uptake, aa_to_exporters_matrix, export_kcats_per_aa, aa_export_kms, ): if mechanistic_uptake: # Export based on mechanistic model trans_counts_per_aa = aa_to_exporters_matrix.astype( np.float64 ) @ aa_transporters_counts.astype(np.float64) export_rates = ( export_kcats_per_aa * trans_counts_per_aa / (1 + aa_export_kms / aa_conc) ) else: # Export is lumped with specific uptake rates in amino_acid_import # and not dependent on internal amino acid concentrations or # explicitly considered here export_rates = np.zeros(len(aa_conc)) return export_rates
[docs] @njit def amino_acid_import_jit( aa_in_media, dry_mass, internal_aa_conc, aa_transporters_counts, mechanisitic_uptake, aa_import_kis, aa_to_importers_matrix, import_kcats_per_aa, max_specific_import_rates, ): saturation = 1 / (1 + internal_aa_conc / aa_import_kis) if mechanisitic_uptake: # Uptake based on mechanistic model counts_per_aa = aa_to_importers_matrix.astype( np.float64 ) @ aa_transporters_counts.astype(np.float64) import_rates = import_kcats_per_aa * counts_per_aa else: import_rates = max_specific_import_rates * dry_mass return import_rates * saturation * aa_in_media
# Class used to update metabolite concentrations based on the current nutrient conditions
[docs] class ConcentrationUpdates(object): def __init__( self, concDict, relative_changes, equilibriumReactions, exchange_data_dict, all_metabolite_ids, linked_metabolites, ): self.units = units.getUnit(list(concDict.values())[0]) self.default_concentrations_dict = dict( (key, concDict[key].asNumber(self.units)) for key in concDict ) self.exchange_fluxes = self._exchange_flux_present(exchange_data_dict) self.relative_changes = relative_changes self._all_metabolite_ids = all_metabolite_ids self.linked_metabolites = linked_metabolites # factor of internal amino acid increase if amino acids present in nutrients self.molecule_scale_factors = { "L-ALPHA-ALANINE[c]": 2.0, "ARG[c]": 2.0, "ASN[c]": 2.0, "L-ASPARTATE[c]": 2.0, "CYS[c]": 2.0, "GLT[c]": 2.0, "GLN[c]": 2.0, "GLY[c]": 2.0, "HIS[c]": 2.0, "ILE[c]": 2.0, "LEU[c]": 2.0, "LYS[c]": 2.0, "MET[c]": 2.0, "PHE[c]": 2.0, "PRO[c]": 2.0, "SER[c]": 2.0, "THR[c]": 2.0, "TRP[c]": 2.0, "TYR[c]": 2.0, "L-SELENOCYSTEINE[c]": 2.0, "VAL[c]": 2.0, } self.molecule_set_amounts = self._add_molecule_amounts( equilibriumReactions, self.default_concentrations_dict ) # return adjustments to concDict based on nutrient conditions
[docs] def concentrations_based_on_nutrients( self, media_id: Optional[str] = None, imports: Optional[str] = None, conversion_units: Optional[units.Unum] = None, ) -> dict[str, Any]: if conversion_units: conversion = self.units.asNumber(conversion_units) else: conversion = self.units if imports is None and media_id is not None: imports = self.exchange_fluxes[media_id] concentrationsDict = self.default_concentrations_dict.copy() metaboliteTargetIds = sorted(concentrationsDict.keys()) concentrations = conversion * np.array( [concentrationsDict[k] for k in metaboliteTargetIds] ) concDict = dict(zip(metaboliteTargetIds, concentrations)) if imports is not None: # For faster conversions than .asNumber(conversion_units) for each setAmount if conversion_units: conversion_to_no_units = conversion_units.asUnit(self.units) # Adjust for measured concentration changes in different media if media_id in self.relative_changes: for mol_id, conc_change in self.relative_changes[media_id].items(): if mol_id in concDict: concDict[mol_id] *= conc_change for moleculeName, setAmount in self.molecule_set_amounts.items(): if ( moleculeName in imports and ( moleculeName[:-3] + "[c]" not in self.molecule_scale_factors or moleculeName == "L-SELENOCYSTEINE[c]" ) ) or ( moleculeName in self.molecule_scale_factors and moleculeName[:-3] + "[p]" in imports ): if conversion_units: setAmount = (setAmount / conversion_to_no_units).asNumber() concDict[moleculeName] = setAmount for met, linked in self.linked_metabolites.items(): concDict[met] = concDict[linked["lead"]] * linked["ratio"] return concDict
[docs] def _exchange_flux_present( self, exchange_data: dict[str, Any] ) -> dict[str, set[str]]: """ Caches the presence of exchanges in each media condition based on exchange_data to set concentrations in concentrations_based_on_nutrients(). Args: exchange_data: dictionary of exchange data for all media conditions with keys:: {importUnconstrainedExchangeMolecules (dict[str, set[str]]): exchange molecules (with location tag) for each media key that do not have an upper bound on their flux, importConstrainedExchangeMolecules (dict[str, dict[str, float with mol/mass/time units]]): constrained molecules (with location tag) for each media key with upper bound flux constraints} Returns: Sets of molecules IDs (with location tags) that can be imported for each media ID """ exchange_fluxes = {} for media, env in exchange_data.items(): exchange_fluxes[media] = {mol for mol, conc in env.items() if conc > 0} return exchange_fluxes
[docs] def _add_molecule_amounts(self, equilibriumReactions, concDict): moleculeSetAmounts = {} for reaction in equilibriumReactions: # We only want to do this for species with standard Michaelis-Menten kinetics initially if len(reaction["stoichiometry"]) != 3: continue moleculeName = [ mol_id for mol_id in reaction["stoichiometry"].keys() if mol_id in self._all_metabolite_ids ][0] amountToSet = 1e-4 moleculeSetAmounts[moleculeName + "[p]"] = amountToSet * self.units moleculeSetAmounts[moleculeName + "[c]"] = amountToSet * self.units for moleculeName, scaleFactor in self.molecule_scale_factors.items(): moleculeSetAmounts[moleculeName] = ( scaleFactor * concDict[moleculeName] * self.units ) return moleculeSetAmounts