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 Oct 1, 2009 021 * Author: Andreas Prlic 022 * 023 */ 024 025package org.biojava.nbio.core.util; 026 027import java.io.ByteArrayInputStream; 028import java.io.File; 029import java.io.FileInputStream; 030import java.io.IOException; 031import java.io.InputStream; 032 033import org.slf4j.Logger; 034import org.slf4j.LoggerFactory; 035 036/** 037 * Provides a cache for storing multiple small files in memory. Can be used to e.g cache gzip compressed PDB files 038 * for avoiding disk IO bottlenecks. 039 * Note this is just a wrapper for the singleton cache. 040 * 041 * @author Andreas Prlic. 042 * 043 */ 044public class FlatFileCache { 045 046 private final static Logger logger = LoggerFactory.getLogger(FlatFileCache.class); 047 048 /** 049 * The cache singleton. 050 */ 051 private static SoftHashMap<String, byte[]> cache = new SoftHashMap<>(0); 052 053 054 // no public constructor; 055 private FlatFileCache(){ 056 057 } 058 059 /** 060 * The file is read and the bytes stored immediately. 061 * <p> 062 * Once added, {@code fileToCache} can be modified or deleted and the cached values will not change. 063 * @param key 064 * @param fileToCache A readable file, of Integer.MAX bytes length or less. 065 */ 066 public static void addToCache(String key, File fileToCache){ 067 //logger.debug("storing " + key + " on file cache (cache size: " + cache.size() + ")"); 068 try (InputStream is = new FileInputStream(fileToCache)){ 069 070 // Get the size of the file 071 long length = fileToCache.length(); 072 073 // You cannot create an array using a long type. 074 // It needs to be an int type. 075 // Before converting to an int type, check 076 // to ensure that file is not larger than Integer.MAX_VALUE. 077 if (length > Integer.MAX_VALUE) { 078 // File is too large 079 throw new IllegalArgumentException("File must be <= " + Integer.MAX_VALUE + " bytes long"); 080 } 081 082 // Create the byte array to hold the data 083 byte[] bytes = new byte[(int)length]; 084 085 // Read in the bytes 086 int offset = 0; 087 int numRead = 0; 088 while (offset < bytes.length 089 && (numRead=is.read(bytes, offset, bytes.length-offset)) >= 0) { 090 offset += numRead; 091 } 092 093 // Ensure all the bytes have been read in 094 if (offset < bytes.length) { 095 is.close(); 096 throw new IOException("Could not completely read file "+fileToCache.getName()); 097 } 098 099 // Close the input stream and return bytes 100 is.close(); 101 102 cache.put(key,bytes); 103 104 } catch (Exception e){ 105 logger.error("Error adding to cache! " + e.getMessage(), e); 106 } 107 } 108 /** 109 * Gets the cached file as an InputStream. 110 * Clients should check for null as the item might have expired in the cache. 111 * @param key 112 * @return An {@code InputStream} or null. 113 */ 114 public static InputStream getInputStream(String key){ 115 //logger.debug("returning " + key + " from file cache (cache size: " + cache.size() + ")"); 116 byte[] bytes = cache.get(key); 117 if ( bytes == null) 118 return null; 119 120 return new ByteArrayInputStream(bytes); 121 122 } 123 124 /** 125 * Returns the number of items in the cache. 126 * If the cache is empty, returns -1 127 * @return 128 */ 129 public static int size() { 130 if ( cache != null) 131 return cache.size(); 132 else 133 return -1; 134 } 135 136 /** 137 * Removes all elements from the cache 138 */ 139 public static void clear(){ 140 cache.clear(); 141 } 142 143}