Stoichiometry with substances on both sides
rcolpo opened this issue · 13 comments
Hi,
I would like to calculate the stoichiometry of reactions like the one bellow:
reac, prod = balance_stoichiometry({'H4P2O7', 'HPO3', 'H2O'}, {'H4P2O7', 'HPO3'})
Answer would be:
H4P2O7 + 4 HPO3 + H2O <-> 2 H4P2O7 + 2 HPO3
OR
2 HPO3 + H2O = H4P2O7
Any idea on how this would be performed? At the moment I receive an error: "Stoichiometry with substances on both sides".
It should be possible to devise an algorithm which comes up with the second solution (which looks preferable to me in the sense that it is arguably more "canoncial"). There is however no such algorithm implemented at the moment.
Thanks bjodah. Any idea on how this could be done?
I would start considering small examples using "pen and paper" first. And solve a bunch of those to develop test cases and get an intuition.
C + CO + CO2 -> C + CO + CO2 # should fail
C + CO + CO2 -> C + CO # suggested solution: C + CO2 -> 2CO
C + CO + CO2 -> C + CO2 # suggested solution: 2CO -> C + CO2
C + CO + CO2 -> CO + CO2 # suggested solution: C + CO2 -> 2CO
C + CO -> C + CO + CO2 # suggested solution: 2CO -> C + CO2
C + CO2 -> C + CO + CO2 # suggested solution: C + CO2 -> 2CO
CO + CO2 -> C + CO + CO2 # suggested solution: 2CO -> C + CO2
here's a start. Do those solutions look like the canonical ones to you?
I think I can throw together a naïve implementation using brute force.
Hi Björn.
Yes, this is exactly the kind of solution I'm looking for. I'm not having much success in implementing a code to replicate these results. Thank you.
@rcolpo I released chempy-0.7.6 with this feature. Make sure you pass underdetermined=None
as well as allow_duplicates=True
.
You are the best. Thank you so much. This will be very useful.
Thanks! (please don't forget to cite the article mentioned in the README if you use it in an academic setting)
I will most certainly do it. Thank you so much.
Hi bjodah,
I receive an error (Failed to remove duplicate keys: ['Mn1']) with the following command:
balance_stoichiometry({'H2O2', 'Mn1', 'H1'}, {'Mn1', 'H2O1'}, allow_duplicates=True, underdetermined=None)
Solution should be:
2 H2O1 = H2O2 + 2 H1
Do you know what the problem is?
Thank you.
I see, here's one possible remedy:
#126
Please take a look at the tests and see if you agree on my choice of expected results.
Thank you so much. This really helped!
Hi bjodah,
Do you think it's also possible to check if a given reaction is already stoichiometrically balanced? (or whether a stoichiometric solution is correct?)
I'm asking this because every reaction has multiple possible stoichiometric solutions, and I would like to test if a suggested solution is valid before trying to find another solution (of course, with repeated metabolites in both sides).
Thank you.
Sure, ReactionSystem
checks for imbalance by default:
>>> ReactionSystem([Reaction({'H2O': 1}, {'H2': 1, 'O2': 1})],
... substances=[Substance.from_formula(f) for f in 'H2O H2 O2'.split()])
---------------------------------------------------------------------------
ValueError Traceback (most recent call last)
<ipython-input-4-82729cf4de88> in <module>()
----> 1 ReactionSystem([Reaction({'H2O': 1}, {'H2': 1, 'O2': 1})], substances=[Substance.from_formula(f) for f in 'H2O H2 O2'.split()])
~/vc/chempy/chempy/reactionsystem.py in __init__(self, rxns, substances, name, checks, substance_factory, missing_substances_from_keys, sort_substances)
102
103 for check in (self.default_checks if checks is None else checks):
--> 104 getattr(self, 'check_'+check)(throw=True)
105
106 if sort_substances:
~/vc/chempy/chempy/reactionsystem.py in check_balance(self, strict, throw)
308 if throw:
309 raise ValueError("Composition violation (%s: %s) in %s" %
--> 310 (k, net, rxn.string(with_param=False)))
311 else:
312 return False
ValueError: Composition violation (8: 1) in H2O -> H2 + O2
(Any improvements to the documentation or the README is of course most welcome)