001package org.biojava.nbio.structure.chem; 002 003import org.biojava.nbio.structure.io.cif.ChemCompConverter; 004import org.slf4j.Logger; 005import org.slf4j.LoggerFactory; 006 007import java.io.BufferedOutputStream; 008import java.io.File; 009import java.io.FileOutputStream; 010import java.io.IOException; 011import java.nio.file.FileSystem; 012import java.nio.file.FileSystems; 013import java.nio.file.Files; 014import java.nio.file.Path; 015import java.nio.file.Paths; 016import java.nio.file.StandardCopyOption; 017import java.util.HashSet; 018import java.util.Set; 019import java.util.zip.ZipEntry; 020import java.util.zip.ZipOutputStream; 021 022/** 023 * This chemical component provider retrieves and caches chemical component definition files from a 024 * zip archive specified in its construction. If the archive does not contain the record, an attempt is 025 * made to download it using DownloadChemCompProvider. The downloaded file is then added to the archive. 026 * 027 * The class is thread-safe and the same ZipChemCompProvider should be used by all threads to prevent 028 * simultaneous read or write to the zip archive. A zip archive will be created if missing. 029 * 030 * @author edlunde 031 * @author larsonm 032 * @since 12/05/12 033 * updated 3/5/2016 for Java 7 ZipFileSystem 034 */ 035public class ZipChemCompProvider implements ChemCompProvider{ 036 private static final Logger s_logger = LoggerFactory.getLogger(ZipChemCompProvider.class); 037 038 private final Path m_tempDir; // Base path where $m_zipRootDir/ will be downloaded to. 039 private final Path m_zipRootDir; 040 private final Path m_zipFile; 041 private final DownloadChemCompProvider m_dlProvider; 042 043 private boolean m_removeCif; 044 045 // Missing IDs from library that cannot be download added here to prevent delays. 046 private Set<String> unavailable = new HashSet<>(); 047 048 /** 049 * ZipChemCompProvider is a Chemical Component provider that stores chemical components 050 * in a zip archive. Missing chemical components are downloaded and appended to the 051 * archive. If non-existent a new zip archive will be created. 052 * 053 * @param chemicalComponentDictionaryFile : path to zip archive for chemical components. 054 * @param tempDir : path for temporary directory, (null) defaults to path in property "java.io.tmpdir". 055 * @throws IOException 056 */ 057 public ZipChemCompProvider(String chemicalComponentDictionaryFile, String tempDir) throws IOException { 058 this.m_zipFile = Paths.get(chemicalComponentDictionaryFile); 059 060 // Use a default temporary directory if not passed a value. 061 if (tempDir == null || "".equals(tempDir)) { 062 this.m_tempDir = Paths.get(System.getProperty("java.io.tmpdir")); 063 } else { 064 this.m_tempDir = Paths.get(tempDir); 065 } 066 067 this.m_zipRootDir = Paths.get("chemcomp"); 068 069 // Setup an instance of the download chemcomp provider. 070 this.m_dlProvider = new DownloadChemCompProvider(m_tempDir.toString()); 071 this.m_removeCif = true; 072 initializeZip(); 073 } 074 075 // See comments in addToZipFileSystem for why initialization is required with 076 // ZipFileSystems - due to URI issues in Java7. 077 private void initializeZip() throws IOException { 078 s_logger.info("Using chemical component dictionary: {}", m_zipFile.toString()); 079 final File f = m_zipFile.toFile(); 080 if (!f.exists()) { 081 s_logger.info("Creating missing zip archive: {}", m_zipFile.toString()); 082 FileOutputStream fo = new FileOutputStream(f); 083 try (ZipOutputStream zip = new ZipOutputStream(new BufferedOutputStream(fo))) { 084 zip.putNextEntry(new ZipEntry("chemcomp/")); 085 zip.closeEntry(); 086 } 087 } 088 } 089 090 /** 091 * Remove downloaded .cif.gz after adding to zip archive? 092 * Default is true. 093 * @param doRemove 094 */ 095 public void setRemoveCif(boolean doRemove) { 096 m_removeCif = doRemove; 097 } 098 099 /** 100 * (non-Javadoc) 101 * @see ChemCompProvider#getChemComp(java.lang.String) 102 * 103 * @param recordName : three letter PDB name for a residue 104 * @return ChemComp from .zip or ChemComp from repository. Will return empty ChemComp when unable to find a residue and will return null if not provided a valid recordName. 105 */ 106 @Override 107 public ChemComp getChemComp(String recordName) { 108 if (null == recordName) return null; 109 110 // handle non-existent ChemComp codes and do not repeatedly attempt to add these. 111 for (String str : unavailable) { 112 if (recordName.equals(str)) return getEmptyChemComp(recordName); 113 } 114 115 // Try to pull from zip, if fail then download. 116 ChemComp cc = getFromZip(recordName); 117 if (cc == null) { 118 s_logger.info("File {} not found in archive. Attempting download from PDB.", recordName); 119 cc = downloadAndAdd(recordName); 120 } 121 122 // If a null record or an empty chemcomp, return a default ChemComp and blacklist. 123 if (cc == null || (null == cc.getName() && cc.getAtoms().size() == 0)) { 124 s_logger.info("Unable to find or download {} - excluding from future searches.", recordName); 125 unavailable.add(recordName); 126 return getEmptyChemComp(recordName); 127 } 128 return cc; 129 } 130 131 /** Use DownloadChemCompProvider to grab a gzipped cif record from the PDB. 132 * Zip all downloaded cif.gz files into the dictionary. 133 * 134 * @param recordName is the three-letter chemical component code (i.e. residue name). 135 * @return ChemComp matching recordName 136 */ 137 private ChemComp downloadAndAdd(String recordName){ 138 final ChemComp cc = m_dlProvider.getChemComp(recordName); 139 140 // final File [] files = finder(m_tempDir.resolve("chemcomp").toString(), "cif.gz"); 141 final File [] files = new File[1]; 142 Path cif = m_tempDir.resolve("chemcomp").resolve(recordName + ".cif.gz"); 143 files[0] = cif.toFile(); 144 if (files[0] != null) { 145 addToZipFileSystem(m_zipFile, files, m_zipRootDir); 146 if (m_removeCif) for (File f : files) f.delete(); 147 } 148 return cc; 149 } 150 151 /** 152 * Cleanup chemical component (.cif.gz) files downloaded to tmpdir. 153 * @param tempdir : path to temporary directory for chemical components 154 */ 155 public static void purgeTempFiles(String tempdir) { 156 if (tempdir == null) return; 157 158 s_logger.info("Removing: "+tempdir); 159 Path dlPath = Paths.get(tempdir).resolve("chemcomp"); 160 File[] chemCompOutFiles = finder(dlPath.toString(), "cif.gz"); 161 if (null != chemCompOutFiles) for (File f : chemCompOutFiles) f.delete(); 162 dlPath.toFile().delete(); 163 } 164 165 /** 166 * Return an empty ChemComp group for a three-letter resName. 167 * @param resName 168 * @return 169 */ 170 private ChemComp getEmptyChemComp(String resName){ 171 String pdbName = ""; // Empty string is default 172 if (null != resName && resName.length() >= 3) { 173 pdbName = resName.substring(0,3); 174 } 175 final ChemComp comp = new ChemComp(); 176 comp.setOneLetterCode("?"); 177 comp.setThreeLetterCode(pdbName); 178 comp.setPolymerType(PolymerType.unknown); 179 comp.setResidueType(ResidueType.atomn); 180 return comp; 181 } 182 183 /** 184 * Return File(s) in dirName that match suffix. 185 * @param dirName 186 * @param suffix 187 * @return 188 */ 189 static private File[] finder(String dirName, final String suffix) { 190 if (null == dirName || null == suffix) { 191 return null; 192 } 193 194 final File dir = new File(dirName); 195 return dir.listFiles((dir1, filename) -> filename.endsWith(suffix)); 196 } 197 198 /** 199 * This is synchronized, along with addToFileSystem to prevent simulatenous reading/writing. 200 * @param recordName to find in zipfile. 201 * @return ChemComp if found or null if missing. 202 */ 203 private synchronized ChemComp getFromZip(String recordName) { 204 ChemComp cc = null; 205 if (!m_zipFile.toFile().exists()) return cc; 206 final String filename = "chemcomp/" + recordName + ".cif.gz"; 207 208 // try with resources block to read from the filesystem. 209 // Don't remove the (ClassLoader) cast! It is required for openjdk 11. 210 try (FileSystem fs = FileSystems.newFileSystem(m_zipFile, (ClassLoader)null)) { 211 Path cif = fs.getPath(filename); 212 213 if (Files.exists(cif)) { 214 s_logger.debug("reading {} from {}", recordName, m_zipFile); 215 final ChemicalComponentDictionary dict = ChemCompConverter.fromPath(cif); 216 cc = dict.getChemComp(recordName); 217 } 218 } catch (IOException e) { 219 s_logger.error("Unable to read from zip file : {}", e.getMessage()); 220 } 221 222 return cc; 223 } 224 225 /** 226 * Add an array of files to a zip archive. 227 * Synchronized to prevent simultaneous reading/writing. 228 * 229 * @param zipFile is a destination zip archive 230 * @param files is an array of files to be added 231 * @param pathWithinArchive is the path within the archive to add files to 232 * @return true if successfully appended these files. 233 */ 234 private synchronized boolean addToZipFileSystem(Path zipFile, File[] files, Path pathWithinArchive) { 235 boolean ret = false; 236 237 /* URIs in Java 7 cannot have spaces, must use Path instead 238 * and so, cannot use the properties map to describe need to create 239 * a new zip archive. ZipChemCompProvider.initilizeZip to creates the 240 * missing zip file */ 241 242 /* 243 // convert the filename to a URI 244 String uriString = "jar:file:" + zipFile.toUri().getPath(); 245 final URI uri = URI.create(uriString); 246 247 // if filesystem doesn't exist, create one. 248 final Map<String, String> env = new HashMap<>(); 249 // Create a new zip if one isn't present. 250 if (!zipFile.toFile().exists()) { 251 System.out.println("Need to create " + zipFile.toString()); 252 } 253 env.put("create", String.valueOf(!zipFile.toFile().exists())); 254 // Specify the encoding as UTF -8 255 env.put("encoding", "UTF-8"); 256 */ 257 258 // Copy in each file. 259 // Don't remove the (ClassLoader) cast! It is required for openjdk 11. 260 try (FileSystem zipfs = FileSystems.newFileSystem(zipFile, (ClassLoader)null)) { 261 Files.createDirectories(pathWithinArchive); 262 for (File f : files) { 263 if (!f.isDirectory() && f.exists()) { 264 Path externalFile = f.toPath(); 265 Path pathInZipFile = zipfs.getPath(pathWithinArchive.resolve(f.getName()).toString()); 266 Files.copy(externalFile, pathInZipFile, 267 StandardCopyOption.REPLACE_EXISTING); 268 } 269 } 270 ret = true; 271 } catch (IOException ex) { 272 s_logger.error("Unable to add entries to Chemical Component zip archive : {}", ex.getMessage()); 273 ret = false; 274 } 275 return ret; 276 } 277}