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

spoon.pattern.PatternBuilder Maven / Gradle / Ivy

/*
 * SPDX-License-Identifier: (MIT OR CECILL-C)
 *
 * Copyright (C) 2006-2019 INRIA and contributors
 *
 * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
 */
package spoon.pattern;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Function;

import spoon.SpoonException;
import spoon.pattern.internal.ValueConvertor;
import spoon.pattern.internal.ValueConvertorImpl;
import spoon.pattern.internal.node.ElementNode;
import spoon.pattern.internal.node.ListOfNodes;
import spoon.pattern.internal.node.RootNode;
import spoon.pattern.internal.parameter.AbstractParameterInfo;
import spoon.pattern.internal.parameter.ParameterInfo;
import spoon.reflect.code.CtBlock;
import spoon.reflect.code.CtStatement;
import spoon.reflect.declaration.CtElement;
import spoon.reflect.declaration.CtType;
import spoon.reflect.declaration.CtVariable;
import spoon.reflect.factory.Factory;
import spoon.reflect.factory.QueryFactory;
import spoon.reflect.path.CtRole;
import spoon.reflect.reference.CtTypeReference;
import spoon.reflect.reference.CtVariableReference;
import spoon.reflect.visitor.Filter;
import spoon.reflect.visitor.chain.CtConsumableFunction;
import spoon.reflect.visitor.chain.CtFunction;
import spoon.reflect.visitor.chain.CtQuery;
import spoon.reflect.visitor.chain.CtQueryable;
import spoon.reflect.visitor.filter.TypeFilter;
import spoon.support.Experimental;

/**
 * The master class to create a {@link Pattern} instance.
 *
 * Main documentation at http://spoon.gforge.inria.fr/pattern.html.
 */
@Experimental
public class PatternBuilder {

	public static final String TARGET_TYPE = "targetType";

	/**
	 * Creates a {@link PatternBuilder} from the List of template elements
	 * @param patternModel a List of Spoon AST nodes, which represents a template of to be generated or to be matched code
	 * @return new instance of {@link PatternBuilder}
	 */
	public static PatternBuilder create(List patternModel) {
		return new PatternBuilder(patternModel);
	}

	public static PatternBuilder create(CtElement... elems) {
		return new PatternBuilder(Arrays.asList(elems));
	}

	private final List patternModel;
	protected final ListOfNodes patternNodes;
	private final Map patternElementToSubstRequests = new IdentityHashMap<>();
	private final Set explicitNodes = Collections.newSetFromMap(new IdentityHashMap<>());

	private CtTypeReference templateTypeRef;
	private final Map parameterInfos = new HashMap<>();
	CtQueryable patternQuery;
	private ValueConvertor valueConvertor;
	private boolean addGeneratedBy = false;
	private boolean autoSimplifySubstitutions = false;
	private boolean built = false;

	static class PatternQuery implements CtQueryable {
		private final QueryFactory queryFactory;
		private final List modelElements;
		PatternQuery(QueryFactory queryFactory, List modelElements) {
			this.queryFactory = queryFactory;
			this.modelElements = modelElements;
		}
		@Override
		public  CtQuery filterChildren(Filter filter) {
			return queryFactory.createQuery(modelElements).filterChildren(filter);
		}
		@Override
		public  CtQuery map(CtFunction function) {
			return queryFactory.createQuery(modelElements).map(function);
		}
		@Override
		public  CtQuery map(CtConsumableFunction queryStep) {
			return queryFactory.createQuery(modelElements).map(queryStep);
		}
	}

	protected PatternBuilder(List template) {
		if (template == null) {
			throw new SpoonException("Cannot create a Pattern from an null model");
		}
		this.templateTypeRef = getDeclaringTypeRef(template);
		this.patternModel = List.copyOf(template);
		this.valueConvertor = new ValueConvertorImpl();
		patternNodes = ElementNode.create(this.patternModel, patternElementToSubstRequests);
		patternQuery = new PatternBuilder.PatternQuery(getFactory().Query(), patternModel);
		if (this.templateTypeRef != null) {
			configurePatternParameters(pb -> {
				pb.parameter(TARGET_TYPE).byType(this.templateTypeRef).setValueType(CtTypeReference.class);
			});
		}
	}

	private CtTypeReference getDeclaringTypeRef(List template) {
		CtType type = null;
		for (CtElement ctElement : template) {
			CtType t;
			if (ctElement instanceof CtType) {
				t = (CtType) ctElement;
				type = mergeType(type, t);
			}

			t = ctElement.getParent(CtType.class);
			if (t != null) {
				type = mergeType(type, t);
			}
		}
		return type == null ? null : type.getReference();
	}

	private CtType mergeType(CtType type, CtType t) {
		if (type == null) {
			return t;
		}
		if (type == t) {
			return type;
		}
		if (type.hasParent(t)) {
			return t;
		}
		if (t.hasParent(type)) {
			return type;
		}
		throw new SpoonException("The pattern on nested types are not supported.");
	}

	/**
	 * @param element a CtElement
	 * @return {@link RootNode}, which handles matching/generation of an `object` from the source spoon AST.
	 * or null, if there is none
	 */
	RootNode getOptionalPatternNode(CtElement element, CtRole... roles) {
		return getPatternNode(true, element, roles);
	}
	RootNode getPatternNode(CtElement element, CtRole... roles) {
		return getPatternNode(false, element, roles);
	}
	private RootNode getPatternNode(boolean optional, CtElement element, CtRole... roles) {
		RootNode node = patternElementToSubstRequests.get(element);
		for (CtRole role : roles) {
			if (node instanceof ElementNode) {
				ElementNode elementNode = (ElementNode) node;
				node = elementNode.getNodeOfRole(role);
				if (node == null) {
					if (optional) {
						return null;
					}
					throw new SpoonException("The role " + role + " resolved to null Node");
				}
			} else {
				if (optional) {
					return null;
				}
				throw new SpoonException("The role " + role + " can't be resolved on Node of class " + node.getClass());
			}
		}
		if (node == null) {
			if (optional) {
				return null;
			}
			throw new SpoonException("There is no Node for element");
		}
		return node;
	}

	void modifyNodeOfElement(CtElement element, ConflictResolutionMode conflictMode, Function elementNodeChanger) {
		RootNode oldNode = patternElementToSubstRequests.get(element);
		RootNode newNode = elementNodeChanger.apply(oldNode);
		if (newNode == null) {
			throw new SpoonException("Removing of Node is not supported");
		}
		handleConflict(conflictMode, oldNode, newNode, tobeUsedNode -> {
			if (!patternNodes.replaceNode(oldNode, tobeUsedNode)) {
				if (conflictMode == ConflictResolutionMode.KEEP_OLD_NODE) {
					//The parent of oldNode was already replaced. OK - Keep that parent old node
					return;
				}
				throw new SpoonException("Old node was not found");
			}
			//update element to node mapping
			patternElementToSubstRequests.put(element, tobeUsedNode);
		});
	}

	void modifyNodeOfAttributeOfElement(CtElement element, CtRole role, ConflictResolutionMode conflictMode, Function elementNodeChanger) {
		modifyNodeOfElement(element, conflictMode, node -> {
			if (node instanceof ElementNode) {
				ElementNode elementNode = (ElementNode) node;
				RootNode oldAttrNode = elementNode.getNodeOfRole(role);
				RootNode newAttrNode = elementNodeChanger.apply(oldAttrNode);
				if (newAttrNode == null) {
					throw new SpoonException("Removing of Node is not supported");
				}
				handleConflict(conflictMode, oldAttrNode, newAttrNode, (tobeUsedNode) -> {
					elementNode.setNodeOfRole(role, tobeUsedNode);
				});
				return node;
			}
			if (conflictMode == ConflictResolutionMode.KEEP_OLD_NODE) {
				return node;
			}
			throw new SpoonException("The Node of atttribute of element cannot be set because element has a Node of class: " + node.getClass().getName());
		});
	}

	private void handleConflict(ConflictResolutionMode conflictMode, RootNode oldNode, RootNode newNode, Consumer applyNewNode) {
		if (oldNode != newNode) {
			if (conflictMode == ConflictResolutionMode.APPEND) {
				if (!(oldNode instanceof ListOfNodes)) {
					oldNode = new ListOfNodes(new ArrayList<>(Arrays.asList(oldNode)));
				}
				if (newNode instanceof ListOfNodes) {
					((ListOfNodes) oldNode).getNodes().addAll(((ListOfNodes) newNode).getNodes());
				} else {
					((ListOfNodes) oldNode).getNodes().add(newNode);
				}
				explicitNodes.add(oldNode);
				explicitNodes.add(newNode);
				applyNewNode.accept(oldNode);
				return;
			}
			if (explicitNodes.contains(oldNode)) {
				//the oldNode was explicitly added before
				if (conflictMode == ConflictResolutionMode.FAIL) {
					throw new SpoonException("Can't replace once assigned Node " + oldNode + " by a " + newNode);
				}
				if (conflictMode == ConflictResolutionMode.KEEP_OLD_NODE) {
					return;
				}
			}
			explicitNodes.remove(oldNode);
			explicitNodes.add(newNode);
			applyNewNode.accept(newNode);
		}
	}

	/**
	 * Changes the Pattern by way the `node` is used at position of template element.
	 * It is used for example to mark that `element` as pattern parameter
	 * @param element
	 * @param node
	 * @param conflictMode
	 */
	void setNodeOfElement(CtElement element, RootNode node, ConflictResolutionMode conflictMode) {
		modifyNodeOfElement(element, conflictMode, oldNode -> {
			return node;
		});
	}

	/**
	 * Changes the Pattern by way the `node` is used at position of template element attribute of `role`.
	 * It is used for example to mark that `element` as pattern parameter
	 * @param element
	 * @param role
	 * @param node
	 * @param conflictMode
	 */
	void setNodeOfAttributeOfElement(CtElement element, CtRole role, RootNode node, ConflictResolutionMode conflictMode) {
		modifyNodeOfAttributeOfElement(element, role, conflictMode, oldAttrNode -> {
			return node;
		});
	}

	/**
	 * @param element to be checked element
	 * @return true if element `element` is a template or a child of template
	 */
	boolean isInModel(CtElement element) {
		if (element != null) {
			for (CtElement patternElement : patternModel) {
				if (element == patternElement || element.hasParent(patternElement)) {
					return true;
				}
			}
		}
		return false;
	}

	/**
	 * Builds a Pattern and returns it
	 * @throws SpoonException if the pattern has been built already
	 * @return the built pattern
	 */
	public Pattern build() {
		if (built) {
			throw new SpoonException("The Pattern may be built only once");
		}
		built = true;
		//clean the mapping so it is not possible to further modify built pattern using this builder
		patternElementToSubstRequests.clear();
		return new Pattern(getFactory(), new ListOfNodes(patternNodes.getNodes())).setAddGeneratedBy(isAddGeneratedBy());
	}

	static List bodyToStatements(CtStatement statementOrBlock) {
		if (statementOrBlock instanceof CtBlock) {
			return ((CtBlock) statementOrBlock).getStatements();
		}
		return Collections.singletonList(statementOrBlock);
	}

	/**
	 * @return default {@link ValueConvertor}, which will be assigned to all new {@link ParameterInfo}s
	 */
	ValueConvertor getDefaultValueConvertor() {
		return valueConvertor;
	}

	/**
	 * @param valueConvertor default {@link ValueConvertor}, which will be assigned to all {@link ParameterInfo}s created after this call
	 * @return this to support fluent API
	 */
	PatternBuilder setDefaultValueConvertor(ValueConvertor valueConvertor) {
		this.valueConvertor = valueConvertor;
		return this;
	}

	/**
	 * All the variable references, whose variables are out of the template model
	 * are automatically marked as pattern parameters
	 * @return this to support fluent API
	 */
	public PatternBuilder configurePatternParameters() {
		configurePatternParameters(pb -> {
			//add this substitution request only if there isn't another one yet
			pb.setConflictResolutionMode(ConflictResolutionMode.KEEP_OLD_NODE);
			/*
			 * detect other parameters.
			 * contract: All variable references, which are declared outside of template are automatically considered as template parameters
			 */
			pb.queryModel().filterChildren(new TypeFilter<>(CtVariableReference.class))
				.forEach((CtVariableReference varRef) -> {
					CtVariable var = varRef.getDeclaration();
					if (var == null || !isInModel(var)) {
						//the varRef has declaration out of the scope of the template. It must be a template parameter.
						ParameterInfo parameter = pb.parameter(varRef.getSimpleName()).getCurrentParameter();
						pb.addSubstitutionRequest(parameter, varRef);
					}
				});
		});
		return this;
	}

	/**
	 * Configure pattern parameters with a {@link PatternParameterConfigurator}
	 * @return this
	 */
	public PatternBuilder configurePatternParameters(Consumer parametersBuilder) {
		PatternParameterConfigurator pb = new PatternParameterConfigurator(this, parameterInfos);
		parametersBuilder.accept(pb);
		return this;
	}

	/**
	 * Used by inline for each statement to define template parameter which is local in the scope of the inline statement
	 */
	PatternBuilder configureLocalParameters(Consumer parametersBuilder) {
		PatternParameterConfigurator pb = new PatternParameterConfigurator(this, new HashMap<>());
		parametersBuilder.accept(pb);
		return this;
	}

	/**
	 * Configures inlined statements
	 *
	 * For example if the `for` statement in this pattern model
	 * 

	 * for(Object x : $iterable$) {
	 *	System.out.println(x);
	 * }
	 * 
* is configured as inline statement and a Pattern is substituted * using parameter $iterable$ = new String[]{"A", "B", "C"} * then pattern generated this code *

	 * System.out.println("A");
	 * System.out.println("B");
	 * System.out.println("C");
	 * 
* because inline statements are executed during substitution process and are not included in generated result. * * The inline statements may be used in PatternMatching process (opposite to Pattern substitution) too. * @param consumer * @return this to support fluent API */ public PatternBuilder configureInlineStatements(Consumer consumer) { InlinedStatementConfigurator sb = new InlinedStatementConfigurator(this); consumer.accept(sb); return this; } static String getLocalTypeRefBySimpleName(CtType templateType, String typeSimpleName) { CtType type = templateType.getNestedType(typeSimpleName); if (type != null) { return type.getQualifiedName(); } type = templateType.getPackage().getType(typeSimpleName); if (type != null) { return type.getQualifiedName(); } Set typeQNames = new HashSet<>(); templateType .filterChildren((CtTypeReference ref) -> typeSimpleName.equals(ref.getSimpleName())) .forEach((CtTypeReference ref) -> typeQNames.add(ref.getQualifiedName())); if (typeQNames.size() > 1) { throw new SpoonException("The type parameter " + typeSimpleName + " is ambiguous. It matches multiple types: " + typeQNames); } if (typeQNames.size() == 1) { return typeQNames.iterator().next(); } return null; } AbstractParameterInfo getParameterInfo(String parameterName) { return parameterInfos.get(parameterName); } protected Factory getFactory() { if (templateTypeRef != null) { return templateTypeRef.getFactory(); } if (!patternModel.isEmpty()) { return patternModel.get(0).getFactory(); } throw new SpoonException("PatternBuilder has no CtElement to provide a Factory"); } /** * @return a {@link CtElement}s which are the template model of this Pattern */ List getPatternModel() { return patternModel; } /** * Calls `consumer` once for each {@link RootNode} element which uses `parameter` * @param parameter to be checked {@link ParameterInfo} * @param consumer receiver of calls */ void forEachNodeOfParameter(ParameterInfo parameter, Consumer consumer) { patternNodes.forEachParameterInfo((paramInfo, vr) -> { if (paramInfo == parameter) { consumer.accept(vr); } }); } /** * @return true if produced Pattern will append generated by comments */ private boolean isAddGeneratedBy() { return addGeneratedBy; } /** * @param addGeneratedBy true when generated by comments have to be appended to each generated type member * @return this to support fluent API */ public PatternBuilder setAddGeneratedBy(boolean addGeneratedBy) { this.addGeneratedBy = addGeneratedBy; return this; } /** * @return true if generated result has to be evaluated to apply simplifications. */ public boolean isAutoSimplifySubstitutions() { return autoSimplifySubstitutions; } /** * @param autoSimplifySubstitutions true if generated result of each substituted has to be evaluated to apply simplifications. * The rule is applied only to substitutions defined after this call * @return this to support fluent API */ public PatternBuilder setAutoSimplifySubstitutions(boolean autoSimplifySubstitutions) { this.autoSimplifySubstitutions = autoSimplifySubstitutions; return this; } CtTypeReference getTemplateTypeRef() { return templateTypeRef; } }