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

com.google.security.fences.FencesMavenEnforcerRule Maven / Gradle / Ivy

Go to download

Augments Java's access control by checking that a Maven Project and all its dependencies conform to a policy that specifies which classes/packages can link to which others.

There is a newer version: 1.9-beta
Show newest version
package com.google.security.fences;

import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.repository.ArtifactRepository;
import org.apache.maven.artifact.resolver.ArtifactNotFoundException;
import org.apache.maven.artifact.resolver.ArtifactResolutionException;
import org.apache.maven.artifact.resolver.ArtifactResolver;
import org.apache.maven.enforcer.rule.api.EnforcerRule;
import org.apache.maven.enforcer.rule.api.EnforcerRuleException;
import org.apache.maven.enforcer.rule.api.EnforcerRuleHelper;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.project.MavenProject;
import org.apache.maven.shared.dependency.tree.DependencyTreeBuilder;
import org.apache.maven.shared.dependency.tree.DependencyTreeBuilderException;
import org.codehaus.plexus.component.configurator.BasicComponentConfigurator;
import org.codehaus.plexus.component.configurator.ComponentConfigurator;
import org.codehaus.plexus.component.configurator.expression.ExpressionEvaluationException;
import org.codehaus.plexus.component.repository.exception.ComponentLookupException;
import org.codehaus.plexus.interpolation.PropertiesBasedValueSource;
import org.w3c.dom.Element;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.security.fences.config.ApiFence;
import com.google.security.fences.config.ClassFence;
import com.google.security.fences.config.Fence;
import com.google.security.fences.config.PackageFence;
import com.google.security.fences.inheritance.InheritanceGraph;
import com.google.security.fences.policy.ApiElement;
import com.google.security.fences.policy.Policy;
import com.google.security.fences.util.LazyString;
import com.google.security.fences.util.Utils;

import java.io.IOException;
import java.io.StringWriter;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;

import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.OutputKeys;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;

/**
 * Augments Java access control by verifying that a project and its dependencies
 * don't statically violate a policy.
 */
public final class FencesMavenEnforcerRule implements EnforcerRule {

  private final List fences = Lists.newArrayList();
  private final LinkedList imports = Lists.newLinkedList();
  private final Set alreadyImported =
      Sets.newLinkedHashSet();

  private void addFence(Fence f) throws EnforcerRuleException {
    f.check();
    fences.add(f);
  }

  /**
   * A setter called by reflection during configuration.  Actually adds
   * instead of blowing away prior value.
   */
  public void setApi(ApiFence x) throws EnforcerRuleException {
    addFence(x);
  }

  /**
   * A setter called by reflection during configuration.  Actually adds
   * instead of blowing away prior value.
   */
  public void setPackage(PackageFence x) throws EnforcerRuleException {
    addFence(x);
  }

  /**
   * A setter called by reflection during configuration.  Actually adds
   * instead of blowing away prior value.
   */
  public void setClass(ClassFence x) throws EnforcerRuleException {
    addFence(x);
  }

  /**
   * A setter called by reflection during configuration.  Actually adds
   * instead of blowing away prior value.
   */
  public void setImport(String x) throws EnforcerRuleException {
    imports.add(new ConfigurationImport(x));
  }

  /**
   * A setter called by reflection during configuration.  Actually adds
   * an {@code } with an {@code } instead of blowing away prior
   * value.
   */
  public void setAddendum(String x) throws EnforcerRuleException {
    Fence api = new ApiFence();
    api.setAddendum(x);
    fences.add(api);
  }

  public void execute(EnforcerRuleHelper helper) throws EnforcerRuleException {
    final Log log = helper.getLog();

    // TODO: maybe check MavenSession.getGoals() to see if this is being
    // run at phase "validate" instead of phase "verify" to warn of a
    // missing verify in the enforcer plugin configuration.

    ArtifactResolver resolver;
    DependencyTreeBuilder treeBuilder;
    ComponentConfigurator configurator;
    try {
      resolver = (ArtifactResolver) helper.getComponent(ArtifactResolver.class);
      treeBuilder = (DependencyTreeBuilder)
          helper.getComponent(DependencyTreeBuilder.class);

      // This seems "the right way" since plexus is supposed to inject
      // dependencies, but when run without -X to turn on debugging,
      // we get a MapOrientedComponentConfigurator which cannot configure
      // this object.
      // http://stackoverflow.com/questions/35919157/using-xmlplexusconfiguration-to-import-more-configuration-for-a-bean-style-maven
      // explains the symptoms.
      //  configurator = (ComponentConfigurator) helper.getComponent(
      //      ComponentConfigurator.class);
      configurator = new BasicComponentConfigurator();
    } catch (ComponentLookupException ex) {
      throw new EnforcerRuleException(
          "Failed to locate component: " + ex.getLocalizedMessage(), ex);
    }

    MavenProject project;
    ArtifactRepository localRepository;
    List remoteRepositories;
    try {
      project = (MavenProject) helper.evaluate("${project}");
      localRepository = (ArtifactRepository)
          helper.evaluate("${localRepository}");
      @SuppressWarnings("unchecked")
      List rr = (List)
          helper.evaluate("${project.remoteArtifactRepositories}");
      remoteRepositories = rr;
    } catch (ExpressionEvaluationException ex) {
      throw new EnforcerRuleException(
          "Failed to locate component: " + ex.getLocalizedMessage(), ex);
    }

    ArtifactFinder finder = new ArtifactFinder(
        resolver, treeBuilder, localRepository, remoteRepositories, log);

    try {
      finder.findClassRoots(project);
    } catch (DependencyTreeBuilderException ex) {
      throw new EnforcerRuleException("Failed to find artifacts", ex);
    } catch (ArtifactResolutionException ex) {
      throw new EnforcerRuleException("Failed to find artifacts", ex);
    } catch (ArtifactNotFoundException ex) {
      throw new EnforcerRuleException("Failed to find artifacts", ex);
    }

    ImmutableList classRoots = finder.getClassRoots();

    InheritanceGraph inheritanceGraph;
    try {
      inheritanceGraph = InheritanceGraphExtractor
          .fromClassRoots(classRoots);
    } catch (IOException ex) {
      throw new EnforcerRuleException(
          "Failed to read classes to find inheritance relationships",
          ex);
    }

    int nAssignedImportOrder = 0;
    int importOrder = 0;
    // Do the imports.
    // Since an import might load a configuration that adds more imports, we
    // just walk the list destructively.
    for (; !imports.isEmpty(); ++importOrder) {
      nAssignedImportOrder = rerootAndAssignImportOrder(
          inheritanceGraph, nAssignedImportOrder, importOrder);
      ConfigurationImport imp = imports.removeFirst();
      if (alreadyImported.add(imp.key)) {
        log.debug("Importing " + imp.key);
        imp.configure(
            this, configurator,
            new ConfigurationImport.ClassRoots(classRoots.iterator()),
            log);
      } else {
        log.info("Not importing " + imp.key + " a second time");
      }
    }
    rerootAndAssignImportOrder(
        inheritanceGraph, nAssignedImportOrder, importOrder);

    ImmutableList allFences = ImmutableList.copyOf(fences);

    if (allFences.isEmpty()) {
      throw new EnforcerRuleException(
          "No fences.  Please configure this rule with a policy."
          + "  See https://github.com/mikesamuel/"
          + "fences-maven-enforcer-rule/blob/master/src/site/markdown/usage.md"
          + " for details");
    }

    // Merge all the fences into one master.
    final ApiFence mergedFence = new ApiFence();
    for (Fence f : allFences) {
      mergedFence.mergeDeep(f);
    }

    // Log the effective configuration
    boolean showConfig = RelevantSystemProperties.shouldShowEffectiveConfig();
    if (showConfig || log.isDebugEnabled()) {
      try {
        Element config = mergedFence.buildEffectiveConfiguration();
        TransformerFactory tf = TransformerFactory.newInstance();
        Transformer transformer = tf.newTransformer();
        transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
        transformer.setOutputProperty(OutputKeys.METHOD, "xml");
        transformer.setOutputProperty(OutputKeys.INDENT, "yes");
        transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8");
        transformer.setOutputProperty(
            "{http://xml.apache.org/xslt}indent-amount", "2");

        StringWriter xmlOut = new StringWriter();
        xmlOut.write("Effective Fences Rule Configuration:\n");
        transformer.transform(
            new DOMSource(config),
            new StreamResult(xmlOut));
        String xml = xmlOut.toString();
        if (showConfig) {
          log.info(xml);
        } else {
          log.debug(xml);
        }
      } catch (ParserConfigurationException ex) {
        log.error(ex);
      } catch (TransformerException ex) {
        log.error(ex);
      }
    }

    checkAllClasses(project, log, inheritanceGraph, mergedFence, classRoots);
  }

  private int rerootAndAssignImportOrder(
      InheritanceGraph inheritanceGraph, int start, int importOrder)
  throws EnforcerRuleException {
    int end = fences.size();
    for (int i = start; i < end; ++i) {
      Fence f = fences.get(i);
      f = f.splitDottedNames(ApiElement.DEFAULT_PACKAGE, inheritanceGraph)
          .promoteToApi();
      f.assignImportOrder(importOrder);
      fences.set(i, f);
    }
    return end;
  }

  protected static void checkAllClasses(
      MavenProject project, Log log, InheritanceGraph inheritanceGraph,
      ApiFence mergedFence, Iterable classRoots)
  throws EnforcerRuleException {
    final Policy p = Policy.fromFence(mergedFence);
    log.debug(new LazyString() {
      @Override
      protected String makeString() {
        return "Using policy\n" + p.toString();
      }
    });

    Checker checker = new Checker(log, inheritanceGraph, p);

    for (ClassRoot classRoot : classRoots) {
      Artifact art = classRoot.art;
      log.info("Checking " + art.getId() + " from scope " + art.getScope());
      try {
        checker.visitAll(ImmutableList.of(classRoot));
      } catch (IOException ex) {
        throw new EnforcerRuleException(
            "Failed to check " + Utils.artToString(art), ex);
      }
    }

    ImmutableList violations = checker.getViolations();
    PolicyViolationReporter reporter = new PolicyViolationReporter(log);
    reporter.interpolator.addValueSource(
        new PropertiesBasedValueSource(project.getProperties()));
    int errorCount = reporter.report(violations);
    if (errorCount != 0) {
      String message = errorCount + " access policy violation"
          + (errorCount == 1 ? "" : "s");
      if (RelevantSystemProperties.inExperimentalMode()) {
        log.info(message + " ignored in experimental mode");
      } else {
        throw new EnforcerRuleException(message);
      }
    }
  }

  public String getCacheId() {
    return null;
  }

  public boolean isCacheable() {
    return false;
  }

  public boolean isResultValid(EnforcerRule arg0) {
    return false;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy