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

com.salesforce.apt.graph.processing.Verifier Maven / Gradle / Ivy

Go to download

This project contains the apt processor that implements all the checks enumerated in @Verify. It is a self contained, and shaded jar.

The newest version!
/*
 * Copyright © 2017, Salesforce.com, Inc
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *     * Redistributions of source code must retain the above copyright
 *       notice, this list of conditions and the following disclaimer.
 *     * Redistributions in binary form must reproduce the above copyright
 *       notice, this list of conditions and the following disclaimer in the
 *       documentation and/or other materials provided with the distribution.
 *     * Neither the name of the  nor the
 *       names of its contributors may be used to endorse or promote products
 *       derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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  BE LIABLE FOR ANY
 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package com.salesforce.apt.graph.processing;

import java.util.Collection;
import java.util.Queue;
import java.util.Set;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.function.Consumer;

import com.salesforce.apt.graph.model.DefinitionModel;
import com.salesforce.apt.graph.model.errors.ErrorModel;
import com.salesforce.apt.graph.model.storage.DefinitionModelStore;
import com.salesforce.apt.graph.types.AssignabilityUtils;

public class Verifier {
  
  public static class ErrorListener implements Consumer {
    
    private ConcurrentLinkedQueue errors = new ConcurrentLinkedQueue<>();
    
    private volatile boolean locked = false;
    
    @Override
    public void accept(ErrorModel t) {
      if (locked) {
        throw new IllegalStateException("This error listener's contents have already been fetched, may not add new errors");
      }
      errors.add(t);
    }

    public boolean hasError() {
      return !errors.isEmpty();
    }
    
    public Queue getErrors() {
      locked = true;
      return errors;
    }
    
  }
  
  /**
   * Verified the definition model.
   * 
   * @param definitions found definitions in this processing
   * @param assignabilityUtils used to determine if the types are assignable (used to verify injection points). 
   * @param store already processed definitions stored as files.
   * @return any errors found during the verification.  Note may not include all possible errors, as short 
   *     circuit logic is necessary in some places.
   */
  public Queue verifyDefinitions(
      Collection definitions, 
      AssignabilityUtils assignabilityUtils,
      DefinitionModelStore store) {
    ErrorListener el = new ErrorListener();
    verifyDefinitions(definitions, store, el, assignabilityUtils);
    return el.getErrors();
  }
  
  /**
   * Verified the definition model.  Registers all errors with the passed in {@link ErrorListener}.
   * 
   * @param definitions found definitions in this processing
   * @param store already processed definitions stored as files.
   * @param el an errorListener which aggregates error information as it is found.
   * @param assignabilityUtils determines if type A is injectable as type B (A is, or is a subclass of B)
   * @return any errors found during the verification.  Note may not include all possible errors, as short 
   *     circuit logic is necessary in some places.
   */
  private void verifyDefinitions(
      Collection definitions,
      DefinitionModelStore store,
      ErrorListener el,
      AssignabilityUtils assignabilityUtils) {

    /*
     * Takes disjoint definitions that have recently been computed from scanning files and links them to each other in
     * to a directed graph of definitions to dependencies.   The supplier will load pre-computed definitions from the 
     * file system. No further processing is needed on the pre-computed and loaded DefinitionModels.
     * 
     * This is stage 2 of the DefinitionModel's life cycle
     */
    new DefinitionJoiner().joinDefinitions(definitions, store, el);

    //short circuit.
    if (el.hasError()) {
      return;
    }
    
    
    
    /*
     * Verifies that the directed graph is actually a directed acyclic graph.
     * 
     * Note that definition heads may not include the full graph of Definitions, pre-computed DefinitionModels will
     * be loaded from files, and will truncate the graph at many locations.
     * 
     * This method doesn't mutate state in any way, aside from accessing data in the definition graph that will cause
     * the will cause the lockedDefintionsMerged flag to be set to true in all definitions passed in, and the 
     * potential generation of errors
     */
    Set definitionHeads = new DefinitionGraphInpector().inspectDefinitionGraph(definitions, el);

    //short circuit.
    if (el.hasError()) {
      return;
    }
  
    /*
     * Check the expected entities are listed and types are correct for supplied object by usage.
     */
    new DefinitionContentInspector().inspectDefinitionGraph(definitionHeads, el, assignabilityUtils, store);

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy