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 Dec 7, 2013 021 * Created by Douglas Myers-Turnbull 022 * 023 */ 024package org.biojava.nbio.structure.io.sifts; 025 026import org.biojava.nbio.structure.align.util.UserConfiguration; 027import org.biojava.nbio.core.sequence.io.util.IOUtils; 028import org.slf4j.Logger; 029import org.slf4j.LoggerFactory; 030 031import java.io.*; 032import java.net.MalformedURLException; 033import java.net.URL; 034import java.util.Collection; 035import java.util.HashMap; 036import java.util.Map; 037import java.util.Map.Entry; 038import java.util.Set; 039import java.util.zip.GZIPInputStream; 040 041/** 042 * A mapping between UniProt entries and PDB chains. 043 * For example 044 * <pre> 045 * SiftsChainToUniprot sifts = SiftsChainToUniprot.load(); 046 * SiftsChainEntry entry1 = sifts.getByUniProtId("P04585"); 047 * System.out.println(entry1.getPdbId() + "." + entry1.getChainName()); // 1hiv.A 048 * System.out.println(entry1.getPdbStart() + "-" + entry1.getPdbStop()); // 1-99 049 * SiftsChainEntry entry2 = sifts.getByChainId("1hiv", "A"); 050 * System.out.println(entry1.equals(entry2)); // true 051 * </pre> 052 * See SIFTS project documentation: https://www.ebi.ac.uk/pdbe/docs/sifts/ 053 * @author dmyersturnbull 054 * @see SiftsChainEntry 055 * @since 3.0.7 056 */ 057public class SiftsChainToUniprotMapping { 058 059 private final static Logger logger = LoggerFactory.getLogger(SiftsChainToUniprotMapping.class); 060 061 062 protected static File DEFAULT_FILE; 063 064 private static final String DEFAULT_FILENAME = "pdb_chain_uniprot.tsv"; 065 private static final URL DEFAULT_URL; 066 067 static { 068 try { 069 DEFAULT_URL = new URL("http://ftp.ebi.ac.uk/pub/databases/msd/sifts/flatfiles/tsv/pdb_chain_uniprot.tsv.gz"); 070 } catch (MalformedURLException e) { 071 throw new RuntimeException(e); 072 } 073 } 074 075 /** 076 * Loads the SIFTS mapping. 077 * Attempts to load the mapping file in the PDB cache directory. 078 * If the file does not exist or could not be parsed, downloads and stores a GZ-compressed file. 079 * @return 080 * @throws IOException If the local file could not be read and could not be downloaded 081 */ 082 public static SiftsChainToUniprotMapping load() throws IOException { 083 return load(false); 084 } 085 086 /** 087 * Loads the SIFTS mapping. 088 * Attempts to load the mapping file in the PDB cache directory. 089 * If the file does not exist or could not be parsed, downloads and stores a GZ-compressed file. 090 * @param useOnlyLocal If true, will throw an IOException if the file needs to be downloaded 091 * @return 092 * @throws IOException If the local file could not be read and could not be downloaded (including if onlyLocal is true) 093 */ 094 public static SiftsChainToUniprotMapping load(boolean useOnlyLocal) throws IOException { 095 096 UserConfiguration config = new UserConfiguration(); 097 File cacheDir = new File(config.getCacheFilePath()); 098 099 DEFAULT_FILE = new File(cacheDir, DEFAULT_FILENAME); 100 101 102 if (!DEFAULT_FILE.exists() || DEFAULT_FILE.length() == 0) { 103 if (useOnlyLocal) throw new IOException(DEFAULT_FILE + " does not exist, and did not download"); 104 download(); 105 } 106 try { 107 return build(); 108 } catch (IOException e) { 109 logger.info("Caught IOException while reading {}. Error: {}",DEFAULT_FILE,e.getMessage()); 110 if (useOnlyLocal) throw new IOException(DEFAULT_FILE + " could not be read, and did not redownload"); 111 download(); 112 return build(); 113 } 114 } 115 116 /** 117 * Builds the mapping by reading SIFTS the tsv file set in {@link #DEFAULT_FILE} variable. 118 * @return 119 * @throws IOException 120 */ 121 protected static SiftsChainToUniprotMapping build() throws IOException { 122 SiftsChainToUniprotMapping sifts = new SiftsChainToUniprotMapping(); 123 BufferedReader br = new BufferedReader(new FileReader(DEFAULT_FILE)); 124 String line = ""; 125 while ((line = br.readLine()) != null) { 126 if (line.isEmpty() || line.startsWith("#") || line.startsWith("PDB")) continue; 127 String[] parts = line.split("\t"); 128 String pdbId = parts[0]; 129 String chainId = parts[1]; 130 String uniProtId = parts[2]; 131 String seqresStart = parts[3]; 132 String seqresEnd = parts[4]; 133 String pdbStart = parts[5]; 134 String pdbEnd = parts[6]; 135 String uniprotStart = parts[7]; 136 String uniprotEnd = parts[8]; 137 SiftsChainEntry entry = new SiftsChainEntry(pdbId, chainId, uniProtId, seqresStart, seqresEnd, 138 pdbStart, pdbEnd, uniprotStart, uniprotEnd); 139 sifts.byChainId.put(pdbId + "." + chainId, entry); 140 sifts.byUniProtId.put(uniProtId, entry); 141 } 142 br.close(); 143 return sifts; 144 } 145 146 private static void download() throws IOException { 147 148 logger.info("Downloading {} to {}",DEFAULT_URL.toString(),DEFAULT_FILE); 149 150 InputStream in = null; 151 OutputStream out = null; 152 153 in = new GZIPInputStream(DEFAULT_URL.openStream()); 154 out = new FileOutputStream(DEFAULT_FILE); 155 IOUtils.copy(in, out); 156 157 } 158 159 private Map<String, SiftsChainEntry> byChainId = new HashMap<>(); 160 161 private Map<String, SiftsChainEntry> byUniProtId = new HashMap<>(); 162 163 private SiftsChainToUniprotMapping() { 164 165 } 166 167 public Set<Entry<String, SiftsChainEntry>> chainEntrySet() { 168 return byChainId.entrySet(); 169 } 170 171 public boolean containsChainId(String pdbId, String chainId) { 172 return byChainId.containsKey(pdbId + "." + chainId); 173 } 174 175 public boolean containsUniProtId(String uniProtId) { 176 return byUniProtId.containsKey(uniProtId); 177 } 178 179 public SiftsChainEntry getByChainId(String pdbId, String chainId) { 180 return byChainId.get(pdbId + "." + chainId); 181 } 182 183 public SiftsChainEntry getByUniProtId(String uniProtId) { 184 return byUniProtId.get(uniProtId); 185 } 186 187 public Set<String> keySet() { 188 return byChainId.keySet(); 189 } 190 191 /** 192 * Returns the number of mapped entries. 193 */ 194 public int size() { 195 return byChainId.size(); 196 } 197 198 public Set<Entry<String, SiftsChainEntry>> uniProtEntrySet() { 199 return byChainId.entrySet(); 200 } 201 202 public Collection<SiftsChainEntry> values() { 203 return byChainId.values(); 204 } 205}