Source code for cobra.test.test_manipulation

# -*- coding: utf-8 -*-
from __future__ import absolute_import

from itertools import chain

import pytest

from cobra.core import Metabolite, Model, Reaction
from cobra.manipulation import *


[docs]class TestManipulation: """Test functions in cobra.manipulation"""
[docs] def test_escape_ids(self, model): model.reactions.PGI.gene_reaction_rule = "a.b or c" assert "a.b" in model.genes escape_ID(model) assert "a.b" not in model.genes
[docs] def test_rename_gene(self, model): original_name = model.genes.b1241.name rename_dict = {"b1241": "foo", "hello": "world", "b3115": "b3115", "b2465": "b3919", "bar": "2935"} modify.rename_genes(model, rename_dict) for i in rename_dict.keys(): if i not in rename_dict.values(): assert i not in model.genes assert "b3115" in model.genes assert "foo" in model.genes assert "world" not in model.genes # make sure the object name was preserved assert model.genes.foo.name == original_name # make sure the reactions are correct assert len(model.genes.foo.reactions) == 2 assert model.reactions.ACALD.gene_reaction_rule == "b0351 or foo" assert model.reactions.TPI.gene_reaction_rule == "b3919" assert model.reactions.TPI.genes == {model.genes.b3919} assert model.reactions.TKT1.gene_reaction_rule == "b2935 or b3919" assert model.reactions.TKT1.genes == { model.genes.b2935, model.genes.b3919} assert model.genes.b3919.reactions == {model.reactions.get_by_id(i) for i in ("TKT1", "TKT2", "TPI")}
[docs] def test_gene_knockout_computation(self, salmonella): def find_gene_knockout_reactions_fast(cobra_model, gene_list): compiled_rules = get_compiled_gene_reaction_rules( cobra_model) return find_gene_knockout_reactions( cobra_model, gene_list, compiled_gene_reaction_rules=compiled_rules) def get_removed(m): return {x.id for x in m._trimmed_reactions} def test_computation(m, gene_ids, expected_reaction_ids): genes = [m.genes.get_by_id(i) for i in gene_ids] expected_reactions = {m.reactions.get_by_id(i) for i in expected_reaction_ids} removed1 = set(find_gene_knockout_reactions(m, genes)) removed2 = set(find_gene_knockout_reactions_fast(m, genes)) assert removed1 == expected_reactions assert removed2 == expected_reactions delete_model_genes(m, gene_ids, cumulative_deletions=False) assert get_removed(m) == expected_reaction_ids undelete_model_genes(m) gene_list = ['STM1067', 'STM0227'] dependent_reactions = {'3HAD121', '3HAD160', '3HAD80', '3HAD140', '3HAD180', '3HAD100', '3HAD181', '3HAD120', '3HAD60', '3HAD141', '3HAD161', 'T2DECAI', '3HAD40'} test_computation(salmonella, gene_list, dependent_reactions) test_computation(salmonella, ['STM4221'], {'PGI'}) test_computation(salmonella, ['STM1746.S'], {'4PEPTabcpp'}) # test cumulative behavior delete_model_genes(salmonella, gene_list[:1]) delete_model_genes(salmonella, gene_list[1:], cumulative_deletions=True) delete_model_genes(salmonella, ["STM4221"], cumulative_deletions=True) dependent_reactions.add('PGI') assert get_removed(salmonella) == dependent_reactions # non-cumulative following cumulative delete_model_genes(salmonella, ["STM4221"], cumulative_deletions=False) assert get_removed(salmonella) == {'PGI'} # make sure on reset that the bounds are correct reset_bound = salmonella.reactions.get_by_id("T2DECAI").upper_bound assert reset_bound == 1000. # test computation when gene name is a subset of another test_model = Model() test_reaction_1 = Reaction("test1") test_reaction_1.gene_reaction_rule = "eggs or (spam and eggspam)" test_model.add_reaction(test_reaction_1) test_computation(test_model, ["eggs"], set()) test_computation(test_model, ["eggs", "spam"], {'test1'}) # test computation with nested boolean expression test_reaction_1.gene_reaction_rule = \ "g1 and g2 and (g3 or g4 or (g5 and g6))" test_computation(test_model, ["g3"], set()) test_computation(test_model, ["g1"], {'test1'}) test_computation(test_model, ["g5"], set()) test_computation(test_model, ["g3", "g4", "g5"], {'test1'}) # test computation when gene names are python expressions test_reaction_1.gene_reaction_rule = "g1 and (for or in)" test_computation(test_model, ["for", "in"], {'test1'}) test_computation(test_model, ["for"], set()) test_reaction_1.gene_reaction_rule = "g1 and g2 and g2.conjugate" test_computation(test_model, ["g2"], {"test1"}) test_computation(test_model, ["g2.conjugate"], {"test1"}) test_reaction_1.gene_reaction_rule = "g1 and (try:' or 'except:1)" test_computation(test_model, ["try:'"], set()) test_computation(test_model, ["try:'", "'except:1"], {"test1"})
[docs] def test_remove_genes(self): m = Model("test") m.add_reactions([Reaction("r" + str(i + 1)) for i in range(8)]) assert len(m.reactions) == 8 rxns = m.reactions rxns.r1.gene_reaction_rule = "(a and b) or (c and a)" rxns.r2.gene_reaction_rule = "(a and b and d and e)" rxns.r3.gene_reaction_rule = "(a and b) or (b and c)" rxns.r4.gene_reaction_rule = "(f and b) or (b and c)" rxns.r5.gene_reaction_rule = "x" rxns.r6.gene_reaction_rule = "y" rxns.r7.gene_reaction_rule = "x or z" rxns.r8.gene_reaction_rule = "" assert "a" in m.genes assert "x" in m.genes remove_genes(m, ["a"], remove_reactions=False) assert "a" not in m.genes assert "x" in m.genes assert rxns.r1.gene_reaction_rule == "" assert rxns.r2.gene_reaction_rule == "" assert rxns.r3.gene_reaction_rule == "b and c" assert rxns.r4.gene_reaction_rule == "(f and b) or (b and c)" assert rxns.r5.gene_reaction_rule == "x" assert rxns.r6.gene_reaction_rule == "y" assert rxns.r7.genes == {m.genes.x, m.genes.z} assert rxns.r8.gene_reaction_rule == "" remove_genes(m, ["x"], remove_reactions=True) assert len(m.reactions) == 7 assert "r5" not in m.reactions assert "x" not in m.genes assert rxns.r1.gene_reaction_rule == "" assert rxns.r2.gene_reaction_rule == "" assert rxns.r3.gene_reaction_rule == "b and c" assert rxns.r4.gene_reaction_rule == "(f and b) or (b and c)" assert rxns.r6.gene_reaction_rule == "y" assert rxns.r7.gene_reaction_rule == "z" assert rxns.r7.genes == {m.genes.z} assert rxns.r8.gene_reaction_rule == ""
[docs] def test_sbo_annotation(self, model): rxns = model.reactions rxns.EX_o2_e.annotation.clear() fake_DM = Reaction("DM_h_c") model.add_reaction(fake_DM) fake_DM.add_metabolites({model.metabolites.get_by_id("h_c"): -1}) # this exchange will be set wrong. The function should not overwrite # an existing SBO annotation rxns.get_by_id("EX_h_e").annotation["sbo"] = "SBO:0000628" add_SBO(model) assert rxns.EX_o2_e.annotation["sbo"] == "SBO:0000627" assert rxns.DM_h_c.annotation["sbo"] == "SBO:0000628" assert rxns.EX_h_e.annotation["sbo"] == "SBO:0000628"
[docs] def test_validate_formula_compartment(self, model): model.metabolites[1].formula = "(a*.bcde)" errors = check_metabolite_compartment_formula(model) assert len(errors) == 1
[docs] def test_validate_mass_balance(self, model): assert len(check_mass_balance(model)) == 0 # if we remove the SBO term which marks the reaction as # mass balanced, then the reaction should be detected as # no longer mass balanced EX_rxn = model.reactions.query(lambda r: r.boundary)[0] EX_rxn.annotation.pop("sbo") balance = check_mass_balance(model) assert len(balance) == 1 assert EX_rxn in balance m1 = Metabolite('m1', formula='()') r1 = Reaction('r1') r1.add_metabolites({m1: 1}) with pytest.raises(ValueError), pytest.warns(UserWarning): r1.check_mass_balance()
[docs] def test_prune_unused_mets_output_type(self, model): # test that the output contains metabolite objects metabolite = model.metabolites.ru5p__D_c [model.reactions.get_by_id(x).remove_from_model() for x in ['RPI', 'RPE', 'GND']] model_pruned, unused = delete.prune_unused_metabolites(model) assert isinstance(model_pruned, Model) assert isinstance(unused[0], Metabolite)
[docs] def test_prune_unused_mets_functionality(self, model): # test that the unused metabolites are not used in the model metabolite1 = model.metabolites.ru5p__D_c metabolite2 = model.metabolites.akg_e metabolite3 = model.metabolites.akg_c reactions = set(chain(metabolite1.reactions, metabolite2.reactions, metabolite3.reactions)) model.remove_reactions(reactions) model_pruned, unused = delete.prune_unused_metabolites(model) assert metabolite1 in model.metabolites assert metabolite2 in model.metabolites assert metabolite3 in model.metabolites assert metabolite1 not in model_pruned.metabolites assert metabolite2 not in model_pruned.metabolites assert metabolite3 not in model_pruned.metabolites
[docs] def test_prune_unused_rxns_output_type(self, model): # test that the output contains reaction objects reaction = Reaction('foo') model.add_reaction(reaction) model_pruned, unused = delete.prune_unused_reactions(model) assert isinstance(model_pruned, Model) assert isinstance(unused[0], Reaction)
[docs] def test_prune_unused_rxns_functionality(self, model): # test that the unused reactions are not used in the model for x in ['foo1', 'foo2', 'foo3']: model.add_reaction(Reaction(x)) model_pruned, unused = delete.prune_unused_reactions(model) assert 'foo1' in model.reactions assert 'foo2' in model.reactions assert 'foo3' in model.reactions assert 'foo1' not in model_pruned.reactions assert 'foo2' not in model_pruned.reactions assert 'foo3' not in model_pruned.reactions