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

com.google.security.fences.classpath.ClassRoot Maven / Gradle / Ivy

package com.google.security.fences.classpath;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipInputStream;

import org.apache.maven.artifact.Artifact;

import com.google.common.base.Preconditions;
import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableMap;
import com.google.security.fences.util.Utils;

/**
 * Encapsulates a bundle of classes that might appear as an entry in a class
 * path.
 */
public final class ClassRoot {
  /** The artifact whose content is at classRoot. */
  public final Artifact art;
  /** Either a JAR (ZIP) file or a class root directory. */
  public final File classRoot;
  /** The kind of class root whether an archive or a root directory. */
  public final ClassRoot.ClassRootKind kind;

  ClassRoot(Artifact art, File classRoot, ClassRoot.ClassRootKind kind) {
    this.art = art;
    this.classRoot = classRoot;
    this.kind = kind;
  }

  /**
   * Reads the relative path, giving an input stream to the given consumer
   * and returning the result of the consumer.
   * This method is responsible for closing the stream.
   */
  
  T readRelativePath(
      String path,
      IOConsumer c)
  throws IOException {
    switch (kind) {
      case ZIPFILE:
        ZipFile zf = new ZipFile(classRoot);
        try {
          ZipEntry e = zf.getEntry(path);
          if (e == null) {
            throw new FileNotFoundException(
                "Could not find " + path + " in " + Utils.artToString(art));
          }
          InputStream is = zf.getInputStream(e);
          try {
            return c.consume(this, path, is);
          } finally {
            is.close();
          }
        } finally {
          zf.close();
        }

      case BUILD_OUTPUT_DIRECTORY:
        File f = classRoot;
        String[] pathElements = path.split("/");
        for (String pathElement : pathElements) {
          if ("".equals(pathElement)) { continue; }
          f = new File(f, pathElement);
        }
        FileInputStream is = new FileInputStream(f);
        try {
          return c.consume(this, path, is);
        } finally {
          is.close();
        }
    }
    throw new AssertionError(kind);
  }

  
  Map readEachPathMatching(
      Predicate relativePathFilter,
      IOConsumer c)
  throws IOException {
    ImmutableMap.Builder b = ImmutableMap.builder();
    switch (kind) {
      case ZIPFILE:
        InputStream in = new FileInputStream(classRoot);
        try {
          ZipInputStream zipIn = new ZipInputStream(in);
          try {
            for (ZipEntry zipEntry;
                 (zipEntry = zipIn.getNextEntry()) != null;) {
              if (!zipEntry.isDirectory()) {
                String entryName = zipEntry.getName();
                if (relativePathFilter.apply(entryName)) {
                  b.put(entryName, c.consume(this, entryName, zipIn));
                }
              }
              zipIn.closeEntry();
            }
          } finally {
            zipIn.close();
          }
        } finally {
          in.close();
        }
        return b.build();

      case BUILD_OUTPUT_DIRECTORY:
        find("", classRoot, relativePathFilter, c, b);
        return b.build();
    }
    throw new AssertionError(kind);
  }

  private  void find(
      String relativePath, File directory,
      Predicate relativePathFilter,
      IOConsumer c,
      ImmutableMap.Builder b)
  throws IOException {
    Preconditions.checkArgument(directory.isDirectory(), directory.getPath());
    File[] contents = directory.listFiles();
    if (contents == null) {
      throw new IOException("Cannot list contents of " + directory);
    } else {
      StringBuilder childRelPathBuilder = new StringBuilder();
      childRelPathBuilder.append(relativePath);
      if (childRelPathBuilder.length() != 0) {
        childRelPathBuilder.append('/');
      }
      int childRelPathPrefixLength = childRelPathBuilder.length();

      for (File child : contents) {
        childRelPathBuilder.setLength(childRelPathPrefixLength);
        String childRelPath = childRelPathBuilder.append(child.getName())
            .toString();
        if (child.isDirectory()) {
          find(childRelPath, child, relativePathFilter, c, b);
        } else if (relativePathFilter.apply(childRelPath)) {
          InputStream in = new FileInputStream(child);
          try {
            T result = c.consume(this, childRelPath, in);
            b.put(childRelPath, result);
          } finally {
            in.close();
          }
        }
      }
    }
  }

  @Override
  public String toString() {
    switch (kind) {
      case ZIPFILE:
        return "zip " + this.classRoot + " from " + art.getId();
      case BUILD_OUTPUT_DIRECTORY:
        return "dir " + this.classRoot + " from " + art.getId();
    }
    throw new AssertionError(kind);
  }

  public enum ClassRootKind {
    ZIPFILE,
    BUILD_OUTPUT_DIRECTORY,
    ;
  }

  interface IOConsumer {
    O consume(ClassRoot root, String relativePath, I x) throws IOException;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy