Verifier.java
- /*
- * 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 <organization> 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 <COPYRIGHT HOLDER> 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<ErrorModel> {
-
- private ConcurrentLinkedQueue<ErrorModel> 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<ErrorModel> 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<ErrorModel> verifyDefinitions(
- Collection<DefinitionModel> 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 supplier 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.
- */
- private void verifyDefinitions(
- Collection<DefinitionModel> definitions,
- DefinitionModelStore store,
- ErrorListener el,
- AssignabilityUtils assignabilityUtils) {
- /*
- * Takes disjoint definitions that have recently been computed from scanning files and compares them against
- * 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<DefinitionModel> 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);
- }
- }