Adduct.java
/*
* Copyright 2021 Dominik Kopczynski, Nils Hoffmann.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.lifstools.jgoslin.domain;
import com.fasterxml.jackson.annotation.JsonIgnore;
import java.util.Map;
import java.util.Map.Entry;
import static java.util.Map.entry;
/**
* An adduct, generally, consists of a sum formula part, an adduct string, the
* charge and the charge sign. An example for a valid adduct is : [M+H]1+.
*
* @author Dominik Kopczynski
* @author Nils Hoffmann
*/
public final class Adduct {
private String sumFormula;
private String adductString;
private int charge;
private int chargeSign;
private ElementTable heavyElements = new ElementTable();
public static final Map<String, ElementTable> ADDUCTS = Map.ofEntries(
entry("+H", ElementTable.of(Map.entry(Element.H, 1))),
entry("+2H", ElementTable.of(Map.entry(Element.H, 2))),
entry("+3H", ElementTable.of(Map.entry(Element.H, 3))),
entry("+4H", ElementTable.of(Map.entry(Element.H, 4))),
entry("-H", ElementTable.of(Map.entry(Element.H, -1))),
entry("-2H", ElementTable.of(Map.entry(Element.H, -2))),
entry("-3H", ElementTable.of(Map.entry(Element.H, -3))),
entry("-4H", ElementTable.of(Map.entry(Element.H, -4))),
entry("+H-H2O", ElementTable.of(Map.entry(Element.H, -1), Map.entry(Element.O, -1))),
entry("+NH4", ElementTable.of(Map.entry(Element.N, 1), Map.entry(Element.H, 4))),
entry("+Cl", ElementTable.of(Map.entry(Element.Cl, 1))),
entry("+HCOO", ElementTable.of(Map.entry(Element.H, 1), Map.entry(Element.C, 1), Map.entry(Element.O, 2))),
entry("+CH3COO", ElementTable.of(Map.entry(Element.H, 3), Map.entry(Element.C, 2), Map.entry(Element.O, 2)))
);
public static final Map<String, Integer> ADDUCT_CHARGES = Map.ofEntries(
entry("+H", 1),
entry("+2H", 2),
entry("+3H", 3),
entry("+4H", 4),
entry("-H", -1),
entry("-2H", -2),
entry("-3H", -3),
entry("-4H", -4),
entry("+H-H2O", 1),
entry("+NH4", 1),
entry("+Cl", -1),
entry("+HCOO", -1),
entry("+CH3COO", -1)
);
public Adduct(String _sum_formula, String _adduct_string) {
this(_sum_formula, _adduct_string, 0, 1);
}
public Adduct(String _sum_formula, String _adduct_string, int _charge, int _sign) {
sumFormula = _sum_formula;
adductString = _adduct_string;
charge = _charge;
setChargeSign(_sign);
}
public void setSumFormula(String sumFormula) {
this.sumFormula = sumFormula;
}
public String getSumFormula() {
return this.sumFormula;
}
public void setAdductString(String adductString) {
this.adductString = adductString;
}
public String getAdductString() {
return this.adductString;
}
public ElementTable getHeavyElements(){
return heavyElements;
}
public void setChargeSign(int _sign) {
if (-1 <= _sign && _sign <= 1) {
chargeSign = _sign;
} else {
throw new ConstraintViolationException("Sign can only be -1, 0, or 1");
}
}
public int getChargeSign() {
return this.chargeSign;
}
public void setCharge(int charge) {
if (charge > 0) {
this.chargeSign = 1;
} else {
this.chargeSign = -1;
}
this.charge = charge;
}
public String getHeavyIsotopeString(){
StringBuilder sb = new StringBuilder();
for (Element e : Elements.ELEMENT_ORDER) {
if (heavyElements.get(e) > 0){
if (heavyElements.get(e) == 1) sb.append(Elements.HEAVY_SHORTCUT.get(e));
else sb.append(Elements.HEAVY_SHORTCUT.get(e)).append(heavyElements.get(e));
}
}
return sb.toString();
}
public String getLipidString() {
if (charge == 0) {
return "[M" +getHeavyIsotopeString() + "]";
}
StringBuilder sb = new StringBuilder();
sb.append("[M").append(getHeavyIsotopeString()).append(sumFormula).append(adductString).append("]").append(charge).append(((chargeSign > 0) ? "+" : "-"));
return sb.toString();
}
@JsonIgnore
public ElementTable getElements() {
ElementTable elements = new ElementTable();
for (Entry<Element, Integer> kv : heavyElements.entrySet()){
if (kv.getValue() > 0){
Element regular = Elements.HEAVY_TO_REGULAR.get(kv.getKey());
elements.put(regular, elements.get(regular) - kv.getValue());
elements.put(kv.getKey(), elements.get(kv.getKey()) + kv.getValue());
}
}
if (adductString.length() > 0){
if (ADDUCTS.containsKey(adductString)) {
if (ADDUCT_CHARGES.get(adductString) != getCharge()) {
throw new ConstraintViolationException("Provided charge '" + getCharge() + "' in contradiction to adduct '" + adductString + "' charge '" + ADDUCT_CHARGES.get(adductString) + "'.");
}
elements.add(ADDUCTS.get(adductString));
} else {
throw new ConstraintViolationException("Adduct '" + adductString + "' is unknown.");
}
}
return elements;
}
/**
* Returns the positive elementary charge times the charge sign.
*
* @return the net charge.
*/
public int getCharge() {
return charge * chargeSign;
}
}