All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ucar.nc2.util.DiskCache Maven / Gradle / Ivy

Go to download

The NetCDF-Java Library is a Java interface to NetCDF files, as well as to many other types of scientific data formats.

There is a newer version: 4.3.22
Show newest version
/*
 * Copyright 1998-2009 University Corporation for Atmospheric Research/Unidata
 *
 * Portions of this software were developed by the Unidata Program at the
 * University Corporation for Atmospheric Research.
 *
 * Access and use of this software shall impose the following obligations
 * and understandings on the user. The user is granted the right, without
 * any fee or cost, to use, copy, modify, alter, enhance and distribute
 * this software, and any derivative works thereof, and its supporting
 * documentation for any purpose whatsoever, provided that this entire
 * notice appears in all copies of the software, derivative works and
 * supporting documentation.  Further, UCAR requests that the user credit
 * UCAR/Unidata in any publications that result from the use of this
 * software or in any product that includes this software. The names UCAR
 * and/or Unidata, however, may not be used in any advertising or publicity
 * to endorse or promote any products or commercial entity unless specific
 * written permission is obtained from UCAR/Unidata. The user also
 * understands that UCAR/Unidata is not obligated to provide the user with
 * any support, consulting, training or assistance of any kind with regard
 * to the use, operation and performance of this software nor to provide
 * the user with any updates, revisions, new versions or "bug fixes."
 *
 * THIS SOFTWARE IS PROVIDED BY UCAR/UNIDATA "AS IS" AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL UCAR/UNIDATA BE LIABLE FOR ANY SPECIAL,
 * INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
 * FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
 * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
 * WITH THE ACCESS, USE OR PERFORMANCE OF THIS SOFTWARE.
 */
package ucar.nc2.util;

import ucar.unidata.util.StringUtil;

import java.io.*;
import java.util.*;
import java.net.URLDecoder;

/**
 * This is a general purpose utility for determining a place to write files and cache them, eg for
 * uncompressing files. This class does not scour itself.
 * 

*

The cdm library sometimes needs to write files, eg * to uncompress them, or for grib index files, etc. The first choice is to write these files * in the same directory that the original file lives in. However, that directory may not be * writeable, so we need to find a place to write them to. * We also want to use the file if it already exists. *

*

A writeable cache "root directory" is set:

    *
  1. explicitly by setRootDirectory() *
  2. through the system property "nj22.cache" if it exists *
  3. by appending "/nj22/cache" to the system property "user.home" if it exists *
  4. by appending "/nj22/cache" to the system property "user.dir" if it exists *
  5. use "./nj22/cache" if none of the above *
*

*

Scenario 1: want to uncompress a file; check to see if already have done so, * otherwise get a File that can be written to. *

 * // see if already uncompressed
 * File uncompressedFile = FileCache.getFile( uncompressedFilename, false);
 * if (!uncompressedFile.exists()) {
 *   // nope, uncompress it
 *   UncompressInputStream.uncompress( uriString, uncompressedFile);
 * }
 * doSomething( uncompressedFile);
 * 
*

*

Scenario 2: want to write a derived file always in the cache. *

 * File derivedFile = FileCache.getCacheFile( derivedFilename);
 * if (!derivedFile.exists()) {
 *   createDerivedFile( derivedFile);
 * }
 * doSomething( derivedFile);
 * 
*

*

Scenario 3: same as scenario 1, but use the default Cache policy: *

 * File wf = FileCache.getFileStandardPolicy( uncompressedFilename);
 * if (!wf.exists()) {
 *   writeToFile( wf);
 *   wf.close();
 * }
 * doSomething( wf);
 * 
* * @author jcaron */ public class DiskCache { static private String root = null; static private boolean standardPolicy = false; static private boolean checkExist = false; /** debug only */ static public boolean simulateUnwritableDir = false; static { root = System.getProperty("nj22.cache"); if (root == null) { String home = System.getProperty("user.home"); if (home == null) home = System.getProperty("user.dir"); if (home == null) home = "."; root = home + "/.unidata/cache/"; } String policy = System.getProperty("nj22.cachePolicy"); if (policy != null) standardPolicy = policy.equalsIgnoreCase("true"); } /** * Set the cache root directory. Create it if it doesnt exist. * * @param cacheDir the cache directory */ static public void setRootDirectory(String cacheDir) { if (!cacheDir.endsWith("/")) cacheDir = cacheDir + "/"; root = StringUtil.replace(cacheDir, '\\', "/"); // no nasty backslash makeRootDirectory(); } /** * Make sure that the current root directory exists. */ static public void makeRootDirectory() { File dir = new File(root); if (!dir.exists()) if (!dir.mkdirs()) throw new IllegalStateException("DiskCache.setRootDirectory(): could not create root directory <" + root + ">."); checkExist = true; } /** * Get the name of the root directory * * @return name of the root directory */ static public String getRootDirectory() { return root; } /** * Set the standard policy used in getWriteableFileStandardPolicy(). * Default is to try to create the file in the same directory as the original. * Setting alwaysInCache to true means to always create it in the cache directory. * * @param alwaysInCache make this the default policy */ static public void setCachePolicy(boolean alwaysInCache) { standardPolicy = alwaysInCache; } /** * Get a File if it exists. If not, get a File that can be written to. * Use the standard policy to decide where to place it. *

* Things are a bit compilicated, because in order to guarentee a file in an arbitrary * location can be written to, we have to try to open it as a FileOutputStream. * If we do, we dont want to open it twice, so we return a WriteableFile that contains an * opened FileOutputStream. * If it already exists, or we get it from cache, we dont need to open it. *

* In any case, you must call WriteableFile.close() to make sure its closed. * * @param fileLocation normal file location * @return WriteableFile holding the writeable File and a possibly opened FileOutputStream (append false) */ static public File getFileStandardPolicy(String fileLocation) { return getFile(fileLocation, standardPolicy); } /** * Get a File if it exists. If not, get a File that can be written to. * If alwaysInCache, look only in the cache, otherwise, look in the "normal" location first, * then in the cache. *

* * @param fileLocation normal file location * @param alwaysInCache true if you want to look only in the cache * @return a File that either exists or is writeable. */ static public File getFile(String fileLocation, boolean alwaysInCache) { if (alwaysInCache) { return getCacheFile(fileLocation); } else { File f = new File(fileLocation); if (f.exists()) return f; // now comes the tricky part to make sure we can open and write to it try { if ( ! simulateUnwritableDir && f.createNewFile()) { f.delete(); return f; } } catch (IOException e) { // cant write to it - drop through } return getCacheFile(fileLocation); } } /** * Get a file in the cache. * File may or may not exist. We assume its always writeable. * If it does exist, set its LastModifiedDate to current time. * * @param fileLocation normal file location * @return equivilent File in the cache. */ static public File getCacheFile(String fileLocation) { File f = new File(makeCachePath(fileLocation)); if (f.exists()) f.setLastModified(System.currentTimeMillis()); if (!checkExist) { File dir = f.getParentFile(); dir.mkdirs(); checkExist = true; } return f; } /** * Make the cache filename * * @param fileLocation normal file location * @return cache filename */ static private String makeCachePath(String fileLocation) { String cachePath; try { fileLocation = fileLocation.replace('\\', '/'); // LOOK - use better normalization code eg Spring StringUtils cachePath = java.net.URLEncoder.encode(fileLocation, "UTF8"); } catch (UnsupportedEncodingException e) { cachePath = java.net.URLEncoder.encode(fileLocation); // shouldnt happen } //String cachePath = StringUtil.replace(fileLocation, '\\', "/"); //cachePath = StringUtil.remove(cachePath, ':'); return root + cachePath; } static public void showCache(PrintStream pw) { pw.println("Cache files"); pw.println("Size LastModified Filename"); File dir = new File(root); for (File file : dir.listFiles()) { String org = null; try { org = URLDecoder.decode(file.getName(), "UTF8"); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } pw.println(" " + file.length() + " " + new Date(file.lastModified()) + " " + org); } } /** * Remove all files with date < cutoff. * * @param cutoff earliest date to allow * @param sbuff write results here, null is ok. */ static public void cleanCache(Date cutoff, StringBuilder sbuff) { if (sbuff != null) sbuff.append("CleanCache files before ").append(cutoff).append("\n"); File dir = new File(root); for (File file : dir.listFiles()) { Date lastMod = new Date(file.lastModified()); if (lastMod.before(cutoff)) { file.delete(); if (sbuff != null) sbuff.append(" delete ").append(file).append(" (").append(lastMod).append(")\n"); } } } /** * Remove files if needed to make cache have less than maxBytes bytes file sizes. * This will remove oldest files first. * * @param maxBytes max number of bytes in cache. * @param sbuff write results here, null is ok. */ static public void cleanCache(long maxBytes, StringBuilder sbuff) { cleanCache(maxBytes, new FileAgeComparator(), sbuff); } /** * Remove files if needed to make cache have less than maxBytes bytes file sizes. * This will remove files in sort order defined by fileComparator. * The first files in the sort order are kept, until the max bytes is exceeded, then they are deleted. * * @param maxBytes max number of bytes in cache. * @param fileComparator sort files first with this * @param sbuff write results here, null is ok. */ static public void cleanCache(long maxBytes, Comparator fileComparator, StringBuilder sbuff) { if (sbuff != null) sbuff.append("DiskCache clean maxBytes= " + maxBytes + "on dir " + root + "\n"); File dir = new File(root); File[] files = dir.listFiles(); List fileList = Arrays.asList(files); Collections.sort(fileList, fileComparator); long total = 0, total_delete = 0; for (File file : fileList) { if (file.length() + total > maxBytes) { total_delete += file.length(); if (sbuff != null) sbuff.append(" delete " + file + " (" + file.length() + ")\n"); file.delete(); } else { total += file.length(); } } if (sbuff != null) { sbuff.append("Total bytes deleted= " + total_delete + "\n"); sbuff.append("Total bytes left in cache= " + total + "\n"); } } static private class FileSizeComparator implements Comparator { public int compare(File f1, File f2) { // return (int) (f1.length() - f2.length()); return (f1.length() { public int compare(File f1, File f2) { //return (int) (f2.lastModified() - f1.lastModified()); return (f1.lastModified()





© 2015 - 2025 Weber Informatics LLC | Privacy Policy