001/** 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 * Created on Feb 22, 2012 021 * Created by Andreas Prlic 022 * 023 * @since 3.0.2 024 */ 025package org.biojava.nbio.structure.io.sifts; 026 027import org.slf4j.Logger; 028import org.slf4j.LoggerFactory; 029import org.w3c.dom.Document; 030import org.w3c.dom.Element; 031import org.w3c.dom.NodeList; 032import org.xml.sax.SAXException; 033 034import javax.xml.parsers.DocumentBuilder; 035import javax.xml.parsers.DocumentBuilderFactory; 036import javax.xml.parsers.ParserConfigurationException; 037 038import java.io.IOException; 039import java.io.InputStream; 040import java.util.ArrayList; 041import java.util.List; 042 043public class SiftsXMLParser { 044 045 private final static Logger logger = LoggerFactory.getLogger(SiftsXMLParser.class); 046 047 048 049 Document dom; 050 List<SiftsEntity> entities; 051 052 static boolean debug = false; 053 public SiftsXMLParser(){ 054 entities = new ArrayList<>(); 055 } 056 057 public List<SiftsEntity> getEntities(){ 058 return entities; 059 } 060 061 062 public void parseXmlFile(InputStream is){ 063 entities = new ArrayList<>(); 064 065 //get the factory 066 DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); 067 068 try { 069 070 //Using factory get an instance of document builder 071 DocumentBuilder db = dbf.newDocumentBuilder(); 072 073 //parse using builder to get DOM representation of the XML file 074 dom = db.parse(is); 075 076 parseDocument(); 077 078 }catch(ParserConfigurationException pce) { 079 pce.printStackTrace(); 080 }catch(SAXException se) { 081 se.printStackTrace(); 082 }catch(IOException ioe) { 083 ioe.printStackTrace(); 084 } 085 } 086 087 088 089 private void parseDocument(){ 090 //get the root element 091 Element docEle = dom.getDocumentElement(); 092 093 //get a nodelist of entities 094 095 NodeList nl = docEle.getElementsByTagName("entity"); 096 if(nl != null && nl.getLength() > 0) { 097 for(int i = 0 ; i < nl.getLength();i++) { 098 099 //get the entity element 100 Element el = (Element)nl.item(i); 101 //get the Employee object 102 SiftsEntity e = getSiftsEntity(el); 103 104 //add it to list 105 entities.add(e); 106 } 107 } 108 } 109 110 /** 111 * <entity type="protein" entityId="A"> 112 */ 113 private SiftsEntity getSiftsEntity(Element empEl) { 114 115 //for each <employee> element get text or int values of 116 //name ,id, age and name 117 118 String type = empEl.getAttribute("type"); 119 String entityId = empEl.getAttribute("entityId"); 120 121 //Create a new Employee with the value read from the xml nodes 122 SiftsEntity entity = new SiftsEntity(type,entityId); 123 124 // get nodelist of segments... 125 NodeList nl = empEl.getElementsByTagName("segment"); 126 if(nl != null && nl.getLength() > 0) { 127 for(int i = 0 ; i < nl.getLength();i++) { 128 129 //get the entity element 130 Element el = (Element)nl.item(i); 131 132 SiftsSegment s = getSiftsSegment(el); 133 134 logger.debug("new segment: {}", s); 135 entity.addSegment(s); 136 137 } 138 } 139 140 logger.debug("new SIFTS entity: {}", entity); 141 return entity; 142 } 143 144 /** segId="4hhb_A_1_140" start="1" end="140" 145 * 146 * @param el 147 * @return 148 */ 149 private SiftsSegment getSiftsSegment(Element el) { 150 151 String segId = el.getAttribute("segId"); 152 String start = el.getAttribute("start"); 153 String end = el.getAttribute("end"); 154 SiftsSegment seg = new SiftsSegment(segId,start,end); 155 156 if ( debug ) 157 System.out.println("parsed " + seg); 158 159 // get nodelist of segments... 160 NodeList nl = el.getElementsByTagName("listResidue"); 161 if(nl != null && nl.getLength() > 0) { 162 for(int i = 0 ; i < nl.getLength();i++) { 163 //get the entity element 164 Element listResidueEl = (Element)nl.item(i); 165 166 NodeList residueNodes = listResidueEl.getElementsByTagName("residue"); 167 if(residueNodes != null && residueNodes.getLength() > 0) { 168 for(int j = 0 ; j < residueNodes.getLength();j++) { 169 Element residue = (Element) residueNodes.item(j); 170 171 SiftsResidue pos = getResidue(residue); 172 seg.addResidue(pos); 173 } 174 } 175 176 } 177 } 178 179 180 return seg; 181 } 182 183 /** 184 * <residue dbResNum="1" dbResName="THR"> 185 <crossRefDb dbSource="PDB" dbVersion="20101103" 186 dbCoordSys="PDBresnum" dbAccessionId="1a4w" dbResNum="1H" 187 dbResName="THR" dbChainId="L"></crossRefDb> 188 <crossRefDb dbSource="UniProt" dbVersion="157-2" 189 dbCoordSys="UniProt" dbAccessionId="P00734" 190 dbResNum="328" dbResName="T"></crossRefDb> 191 <crossRefDb dbSource="SCOP" dbVersion="1.75" 192 dbCoordSys="PDBresnum" dbAccessionId="26083" 193 dbResNum="1H" dbResName="THR" dbChainId="L"></crossRefDb> 194 <residueDetail dbSource="MSD" property="Annotation"> 195 Not_Observed</residueDetail> 196 </residue> 197 198 */ 199 private SiftsResidue getResidue(Element residue) { 200 201 SiftsResidue res = new SiftsResidue(); 202 203 String dbResNumS = residue.getAttribute("dbResNum"); 204 res.setNaturalPos(Integer.parseInt(dbResNumS)); 205 206 String seqResName = residue.getAttribute("dbResName"); 207 res.setSeqResName(seqResName); 208 209 boolean observed = true; 210 211 List<String> details = getTextValues(residue, "residueDetail"); 212 213 if ( details != null && details.contains("Not_Observed")){ 214 observed = false; 215 } 216 res.setNotObserved(! observed); 217 //else if ( detail != null && detail.trim().equalsIgnoreCase("Conflict")){ 218 // 219 //} 220 221 NodeList nl = residue.getElementsByTagName("crossRefDb"); 222 if(nl != null && nl.getLength() > 0) { 223 for(int i = 0 ; i < nl.getLength();i++) { 224 //get the entity element 225 Element crossRefEl = (Element)nl.item(i); 226 227 String dbSource = crossRefEl.getAttribute("dbSource"); 228 String dbCoordSys = crossRefEl.getAttribute("dbCoordSys"); 229 String dbAccessionId = crossRefEl.getAttribute("dbAccessionId"); 230 String dbResNum = crossRefEl.getAttribute("dbResNum"); 231 String dbResName = crossRefEl.getAttribute("dbResName"); 232 String dbChainId = crossRefEl.getAttribute("dbChainId"); 233 234 // System.out.println(dbSource + " " + dbCoordSys + " " + dbAccessionId + " " + dbResNum + " " + dbResName + " " + dbChainId); 235 236 if ( "PDB".equals(dbSource) && "PDBresnum".equals(dbCoordSys)){ 237 res.setPdbResNum(dbResNum); 238 res.setPdbResName(dbResName); 239 res.setChainId(dbChainId); 240 res.setPdbId(dbAccessionId); 241 } else if ( "UniProt".equals(dbSource)){ 242 res.setUniProtPos(Integer.parseInt(dbResNum)); 243 res.setUniProtResName(dbResName); 244 res.setUniProtAccessionId(dbAccessionId); 245 } 246 } 247 } 248 return res; 249 } 250 251 252 253 /** 254 * I take a xml element and the tag name, look for the tag and get 255 * the text content 256 * i.e for <employee><name>John</name></employee> xml snippet if 257 * the Element points to employee node and tagName is 'name' I will return John 258 */ 259 @SuppressWarnings("unused") 260 private String getTextValue(Element ele, String tagName) { 261 String textVal = null; 262 NodeList nl = ele.getElementsByTagName(tagName); 263 if(nl != null && nl.getLength() > 0) { 264 Element el = (Element)nl.item(0); 265 textVal = el.getFirstChild().getNodeValue(); 266 } 267 268 return textVal; 269 } 270 271 private List<String> getTextValues(Element ele, String tagName) { 272 List<String>values = new ArrayList<>(); 273 NodeList nl = ele.getElementsByTagName(tagName); 274 if(nl != null && nl.getLength() > 0) { 275 for ( int i = 0 ;i < nl.getLength() ; i ++) { 276 277 Element n = (Element) nl.item(i); 278 279 @SuppressWarnings("unused") 280 String k = n.getNodeName(); 281 282 String val = n.getFirstChild().getNodeValue(); 283 if ( val != null) 284 values.add(val); 285 } 286 } 287 288 return values; 289 } 290 291 292 293 294 295 296 297 }