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

org.armedbear.lisp.ZipCache Maven / Gradle / Ivy

There is a newer version: 1.9.2
Show newest version
/*
 * ZipCache.java
 *
 * Copyright (C) 2010, 2014 Mark Evenson
 * $Id$
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 *
 * As a special exception, the copyright holders of this library give you
 * permission to link this library with independent modules to produce an
 * executable, regardless of the license terms of these independent
 * modules, and to copy and distribute the resulting executable under
 * terms of your choice, provided that you also meet, for each linked
 * independent module, the terms and conditions of the license of that
 * module.  An independent module is a module which is not derived from
 * or based on this library.  If you modify this library, you may extend
 * this exception to your version of the library, but you are not
 * obligated to do so.  If you do not wish to do so, delete this
 * exception statement from your version.
 */
package org.armedbear.lisp;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import org.armedbear.lisp.util.HttpHead;
import static org.armedbear.lisp.Lisp.*;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.JarURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.text.ParsePosition;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.HashMap;
import java.util.Locale;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.zip.ZipException;
import java.util.zip.ZipFile;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

/**
 * A cache for all zip/jar file access by JarPathname that uses the last
 * modified time of the cached resource.
 *
 * If you run into problems with caching, use
 * (SYS::DISABLE-ZIP-CACHE).  Once disabled, the caching cannot be
 * re-enabled.
 *
 */ 
public class ZipCache {
  public static final boolean checkZipFile(Pathname name) {
    InputStream input = name.getInputStream();
    try {
      byte[] bytes = new byte[4];
      int bytesRead = input.read(bytes);
      return bytesRead == 4 && bytes[0] == 80 && bytes[1] == 75 && bytes[2] == 3 && bytes[3] == 4;
    } catch (Throwable t) {
      // any error probably means 'no'
      return false;
    } finally {
      if (input != null) {
        try {
          input.close();
        } catch (IOException e) {
        } // ignore exceptions
      }
    }
  }
  static InputStream getInputStream(ZipFile jarFile, String entryPath) {
    ZipEntry entry = jarFile.getEntry(entryPath);
    if (entry == null) {
      Debug.trace("Failed to find entry " + "'" + entryPath + "'" + " in " + "'" + jarFile.getName() + "'");
      return null;
    }
    InputStream result = null;
    try {
      result = jarFile.getInputStream(entry);
    } catch (IOException e) {
      Debug.trace("Failed to open InputStream for " + "'" + entryPath + "'" + " in " + "'" + jarFile.getName() + "'");
      return null;
    }
    return result;
  }
  public static ZipInputStream getZipInputStream(ZipFile zipfile, String entryName) {
    return ZipCache.getZipInputStream(zipfile, entryName, false);
  }
  public static ZipInputStream getZipInputStream(ZipFile zipfile, String entryName, boolean errorOnFailure) {
    ZipEntry zipEntry = zipfile.getEntry(entryName);
    ZipInputStream stream = null;
    try {
      stream = new ZipInputStream(zipfile.getInputStream(zipEntry));
    } catch (IOException e) {
      if (errorOnFailure) {
        simple_error("Failed to open '" + entryName + "' in zipfile '" + zipfile + "': " + e.getMessage());
      }
      return null;
    }
    return stream;
  }
  public static ByteArrayOutputStream readEntry(ZipInputStream stream) {
    ByteArrayOutputStream result = new ByteArrayOutputStream();
    int count;
    byte[] buf = new byte[1024]; // What's a decent buffer size?
    try {
      while ((count = stream.read(buf, 0, buf.length)) != -1) {
        result.write(buf, 0, count);
      }
    } catch (IOException e) {
      Debug.trace("Failed to read entry from " + stream + ": " + e);
      return null;
    }
    return result;
  }
  public static ZipEntry getEntry(ZipInputStream zipInputStream, String entryName) {
    return ZipCache.getEntry(zipInputStream, entryName, false);
  }
  public static ZipEntry getEntry(ZipInputStream zipInputStream, String entryName, boolean errorOnFailure) {
    ZipEntry entry = null;
    do {
      try {
        entry = zipInputStream.getNextEntry();
      } catch (IOException e) {
        if (errorOnFailure) {
          Lisp.error(new FileError("Failed to seek for " + "'" + entryName + "'" + " in " + zipInputStream.toString()));
        }
        return null;
      }
    } while (entry != null && !entry.getName().equals(entryName));
    if (entry != null) {
      return entry;
    }
    if (errorOnFailure) {
      Lisp.error(new FileError("Failed to find " + "'" + entryName + "'" + " in " + zipInputStream.toString()));
    }
    return null;
  }

  public static InputStream getEntryAsInputStream(ZipInputStream zipInputStream, String entryName) {
    ZipEntry entry = getEntry(zipInputStream, entryName);
    ByteArrayOutputStream bytes = readEntry(zipInputStream);
    return new ByteArrayInputStream(bytes.toByteArray());
  }

  public static InputStream getEntryAsInputStream(JarPathname archiveEntry) {
    JarPathname archiveJar = archiveEntry.getArchive();
    Archive archive = ZipCache.getArchive(archiveJar);
    InputStream result = archive.getEntryAsInputStream(archiveEntry);
    if (result == null) {
      simple_error("Failed to get InputStream for ~a", archiveEntry);
    }
    return result;
  }

  // To make this thread safe, we should return a proxy for ZipFile
  // that keeps track of the number of outstanding references handed
  // out, not allowing ZipFile.close() to succeed until that count
  // has been reduced to 1 or the finalizer is executing.
  // Unfortunately the relatively simple strategy of extending
  // ZipFile via a CachedZipFile does not work because there is not
  // a null arg constructor for ZipFile.
  static HashMap cache = new HashMap();

  abstract static public class Archive {
    JarPathname root;
    LinkedHashMap entries
      = new LinkedHashMap();
    long lastModified;

    abstract InputStream getEntryAsInputStream(JarPathname entry);
    abstract ZipEntry getEntry(JarPathname entry);
    abstract void populateAllEntries();
    abstract void close();
  }

  static public class ArchiveStream
    extends Archive
  {
    ZipInputStream source;
    ZipEntry rootEntry;

    public ArchiveStream(InputStream stream, JarPathname root, ZipEntry rootEntry) {
      if (!(stream instanceof ZipInputStream)) {
        this.source = new ZipInputStream(stream);
      } else {
        this.source = (ZipInputStream)stream;
      }
      this.root = root;
      this.rootEntry = rootEntry;
      this.lastModified = rootEntry.getTime(); // FIXME how to re-check time as modified?
    }  

    // TODO wrap in a weak reference to allow JVM to possibly reclaim memory
    LinkedHashMap contents
      = new LinkedHashMap();

    boolean populated = false;

    public InputStream getEntryAsInputStream(JarPathname entry) {
      if (!populated) {
        populateAllEntries();
      }

      ByteArrayOutputStream bytes = contents.get(entry);
      if (bytes != null) {
        return new ByteArrayInputStream(bytes.toByteArray());
      }
      return null;
    }

    public ZipEntry getEntry(JarPathname entry) {
      if (!populated) {
        populateAllEntries();
      }
      ZipEntry result = entries.get(entry);
      return result;
    }

    void populateAllEntries() {
      if (populated) {
        return;
      }
      ZipEntry entry;
      try {
        while ((entry = source.getNextEntry()) != null) {
          String name = entry.getName();
          JarPathname entryPathname
            = (JarPathname)JarPathname.createEntryFromJar(root, name);
          entries.put(entryPathname, entry);
          ByteArrayOutputStream bytes
            = readEntry(source);
          contents.put(entryPathname, bytes);
        }
        populated = true;
      } catch (IOException e) {
        simple_error("Failed to read entries from zip archive", root);
      }
    }

    void close () {
      if (source != null) {
        try {
          source.close();
        } catch (IOException ex) {
          {}
        }
      }
    }
  }

  static public class ArchiveURL
    extends ArchiveFile
  {
    JarURLConnection connection;

    public ArchiveURL(JarPathname jar)
      throws java.io.IOException
    {
      String rootJarURLString = jar.getRootJarAsURLString();
      URL rootJarURL = new URL(rootJarURLString);
      JarURLConnection jarConnection
        = (JarURLConnection) rootJarURL.openConnection();

      this.root = jar;
      this.connection = jarConnection;
      this.file = (ZipFile)connection.getJarFile();
      this.lastModified = connection.getLastModified();
    }
    
    void close() {
      super.close();
      // TODO: do we need to clean up from the connection?
    }
  }

  static public class ArchiveFile
    extends Archive
  { 
    ZipFile file;

    ZipFile get() { return file;}

    ArchiveFile() {}

    public ArchiveFile(JarPathname jar)
      throws ZipException, IOException
    {
      File f = ((Pathname)jar.getRootJar()).getFile();
      this.root = jar;
      this.file = new ZipFile(f);
      this.lastModified = f.lastModified();
    }

    public ZipEntry getEntry(JarPathname entryPathname) {
      ZipEntry result = entries.get(entryPathname);
      if (result != null) {
        return result;
      }
      String entryPath = entryPathname.asEntryPath();
      result = file.getEntry(entryPath);

      if (result == null) {
        return null;
      }

      // ZipFile.getEntry() will return directories when asked for
      // files.
      if (result.isDirectory()
          && (!entryPathname.getName().equals(NIL)
              || !entryPathname.getType().equals(NIL))) {
        return null;
      }

      entries.put(entryPathname, result);
      return result;
    }

    void populateAllEntries() {
      ZipFile f = file;
      if (f.size() == entries.size()) {
        return;
      }

      Enumeration e = f.entries();
      while (e.hasMoreElements()) {
        ZipEntry entry = e.nextElement();
        String name = entry.getName();
        JarPathname entryPathname
          = (JarPathname)JarPathname.createEntryFromJar(root, name);
        entries.put(entryPathname, entry);
      }
    }

    InputStream getEntryAsInputStream(JarPathname entry) {
      InputStream result = null;
      ZipEntry zipEntry = getEntry(entry);

      try { 
        result = file.getInputStream(zipEntry);
      } catch (IOException e) {} // FIXME how to signal a meaningful error?

      return result;
    }
    void close() {
      if (file != null) {
        try {
          file.close();
        } catch (IOException e) {}
          
      }
    }
  }

  static boolean cacheEnabled = true;
  private final static Primitive DISABLE_ZIP_CACHE = new disable_zip_cache();
  final static class disable_zip_cache extends Primitive {
    disable_zip_cache() {
      super("disable-zip-cache", PACKAGE_SYS, true, "",
            "Disable all caching of ABCL FASLs and ZIPs.");
    }
    @Override
    public LispObject execute() {
      ZipCache.disable();
      return T;
    }
  }
  static public synchronized void disable() {
    cacheEnabled = false;
    cache.clear();  
  }

  synchronized public static LinkedHashMap getEntries(JarPathname jar) {
    Archive archive = getArchive(jar);
    archive.populateAllEntries(); // Very expensive for jars with large number of entries
    return archive.entries;
  }

  synchronized public static Iterator> getEntriesIterator(JarPathname jar) {
    LinkedHashMap entries = getEntries(jar);
    Set> set = entries.entrySet();
    return set.iterator();
  }

  static ZipEntry getZipEntry(JarPathname archiveEntry) {
    JarPathname archiveJar = archiveEntry.getArchive();
    Archive zip = getArchive(archiveJar);
    ZipEntry entry = zip.getEntry(archiveEntry);
    return entry;
  }

  // ??? we assume that DIRECTORY, NAME, and TYPE components are NIL
  synchronized public static Archive getArchive(JarPathname jar) {
    Archive result = cache.get(jar);
    if (result != null) {
      return result;
    }
    Pathname rootJar = (Pathname) jar.getRootJar();
    LispObject innerJars = jar.getJars().cdr();

    if (!rootJar.isLocalFile()) {
      return getArchiveURL(jar);
    }
    
    if (innerJars.equals(NIL)) {
      return getArchiveFile(jar);
    } 

    result = getArchiveStreamFromFile(jar);
    cache.put(result.root, result); 

    JarPathname nextArchive = new JarPathname();
    nextArchive
      .setDevice(new Cons(rootJar,
                          new Cons(innerJars.car(), NIL)))
      .setDirectory(NIL)
      .setName(NIL)
      .setType(NIL);
      
    innerJars = innerJars.cdr();
    while (innerJars.car() != NIL) {
      Pathname nextJarArchive = (Pathname)innerJars.car();
      
      JarPathname nextAsEntry = new JarPathname();
      nextAsEntry
        .setDevice(nextArchive.getDevice())
        .setDirectory(nextJarArchive.getDirectory())
        .setName(nextJarArchive.getName())
        .setType(nextJarArchive.getType());
      // FIXME
      // The pathnames for subsquent entries in a PATHNAME-JAR
      // are relative.  Should they be?
      LispObject directories = nextAsEntry.getDirectory();
      if ( !directories.equals(NIL)
           && directories.car().equals(Keyword.RELATIVE)) {
        directories = directories.cdr().push(Keyword.ABSOLUTE);
        nextAsEntry.setDirectory(directories);
      }

      nextArchive.setDevice(nextArchive.getDevice().reverse().push(nextJarArchive).reverse());
      ArchiveStream stream = (ArchiveStream) result;

      ZipEntry entry = stream.getEntry(nextAsEntry);
      if (entry == null) {
        return null;
      }
      
      InputStream inputStream = stream.getEntryAsInputStream(nextAsEntry);
      if (inputStream == null) {
        return null;
      }
      stream = new ArchiveStream(inputStream, nextArchive, entry);
      result = stream;
      cache.put(nextArchive, result); 

      innerJars = innerJars.cdr();
      if (innerJars.cdr().equals(NIL)
          && (!jar.getDirectory().equals(NIL)
              && jar.getName().equals(NIL)
              && jar.getType().equals(NIL))) {
        simple_error("Currently unimplemented retrieval of an entry in a nested pathnames");
        return (Archive)UNREACHED;
      }
    }
    return result;
  }

  static ArchiveStream getArchiveStreamFromFile(JarPathname p) {
      JarPathname root = new JarPathname();
      root.setDevice(new Cons(p.getRootJar(), NIL));
      Pathname nextJar = (Pathname)p.getJars().cdr().car();      
      ArchiveFile rootArchiveFile = (ArchiveFile)getArchiveFile(root);

      JarPathname innerArchive = new JarPathname();

      LispObject jars = list(p.getRootJar(), nextJar);
      innerArchive.setDevice(jars);

      JarPathname innerArchiveAsEntry = new JarPathname();
      innerArchiveAsEntry
        .setDevice(new Cons(root, NIL))
        .setDirectory(nextJar.getDirectory())
        .setName(nextJar.getName())
        .setType(nextJar.getType());

      ZipEntry entry = rootArchiveFile.getEntry(innerArchiveAsEntry);
      if (entry == null) {
        return null;
      }
      InputStream inputStream = rootArchiveFile.getEntryAsInputStream(innerArchiveAsEntry);
      if (inputStream == null) {
        return null;
      }
      ArchiveStream result = new ArchiveStream(inputStream, innerArchive, entry);
      return result;
  }
    
  

  public static Archive getArchiveURL(JarPathname jar) {
    Pathname rootJar = (Pathname) jar.getRootJar();

    URL rootJarURL = null;
    try {
      ArchiveURL result = new ArchiveURL(jar);
      cache.put(jar, result);
      return result;
    } catch (MalformedURLException e) {
      simple_error("Failed to form root URL for ~a", jar);
      return (Archive)UNREACHED;      
    } catch (IOException e) {
      simple_error("Failed to fetch ~a: ~a", jar, e);
      return (Archive)UNREACHED;      
    }
  }

  static public Archive getArchiveFile(JarPathname jar) {
    try {
      ArchiveFile result = new ArchiveFile(jar);
      cache.put(jar, result);
      return result;
    } catch (ZipException e) {
      error(new FileError("Failed to open local zip archive"
                          + " because " + e, jar));
                          
      return (Archive)UNREACHED;
    } catch (IOException e) {
      error(new FileError("Failed to open local zip archive"
                          + " because " + e, jar));
      return (Archive)UNREACHED;
    }
  }

  // unused
  static void checkRemoteLastModified(ArchiveURL archive) {
    // Unfortunately, the Apple JDK under OS X doesn't do
    // HTTP HEAD requests, instead refetching the entire
    // resource, and I assume this is the case in all
    // Sun-derived JVMs.  So, we use a custom HEAD
    // implementation only looking for Last-Modified
    // headers, which if we don't find, we give up and
    // refetch the resource.

    String dateString = null;

    String url = archive.root.getRootJarAsURLString();
    
    try {
      dateString = HttpHead.get(url, "Last-Modified");
    } catch (IOException ex) {
      Debug.trace(ex);
    }
    Date date = null;
    ParsePosition pos = new ParsePosition(0);

    final SimpleDateFormat ASCTIME
      = new SimpleDateFormat("EEE MMM d HH:mm:ss yyyy", Locale.US);
    final SimpleDateFormat RFC_1036
      = new SimpleDateFormat("EEEE, dd-MMM-yy HH:mm:ss zzz", Locale.US);
    final SimpleDateFormat RFC_1123
      = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss zzz", Locale.US);

    if (dateString != null) {
      date = RFC_1123.parse(dateString, pos);
      if (date == null) {
        date = RFC_1036.parse(dateString, pos);
        if (date == null) {
          date = ASCTIME.parse(dateString, pos);
        }
      }
    }

    // Replace older item in cache
    if (date == null || date.getTime() > archive.lastModified) {
      JarPathname root = archive.root;
      Archive entry = getArchiveURL(root);
      cache.put(root, entry);
    }
    if (date == null) {
      if (dateString == null) {
        Debug.trace("Failed to retrieve request header: "
                    + url.toString());
      } else {
        Debug.trace("Failed to parse Last-Modified date: " +
                    dateString);
      }
    }
  }

  // ## clear-zip-cache  => boolean
  private static final Primitive CLEAR_ZIP_CACHE = new clear_zip_cache();
  private static class clear_zip_cache extends Primitive { 
    clear_zip_cache() {
      super("clear-zip-cache", PACKAGE_SYS, true);
    }
    @Override
    public LispObject execute() {
      int size = cache.size();
      cache.clear();
      return size == 0 ? NIL : T;
    }
  }

  // ## remove-zip-cache-entry pathname => boolean
  private static final Primitive REMOVE_ZIP_CACHE_ENTRY = new remove_zip_cache_entry();
  private static class remove_zip_cache_entry extends Primitive { 
    remove_zip_cache_entry() {
      super("remove-zip-cache-entry", PACKAGE_SYS, true, "pathname");
    }
    @Override
    public LispObject execute(LispObject arg) {
      Pathname p = coerceToPathname(arg);
      boolean result = false;
      if (p instanceof JarPathname) {
        result = ZipCache.remove((JarPathname)p);
      } 
      return result ? T : NIL;
    }
  }

  synchronized public static boolean remove(Pathname pathname) {
    JarPathname p = JarPathname.createFromPathname(pathname);
    return remove(p);
  }
      
  synchronized public static boolean remove(JarPathname p) {
    Archive archive = cache.get(p);
    if (archive != null) {
      archive.close();
      cache.remove(p);
      return true;
    }
    return false;
  }
}





© 2015 - 2024 Weber Informatics LLC | Privacy Policy