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

spoon.refactoring.Refactoring Maven / Gradle / Ivy

Go to download

Spoon is a tool for meta-programming, analysis and transformation of Java programs.

There is a newer version: 11.1.1-beta-14
Show newest version
/*
 * SPDX-License-Identifier: (MIT OR CECILL-C)
 *
 * Copyright (C) 2006-2023 INRIA and contributors
 *
 * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) or 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.refactoring;

import spoon.SpoonException;
import spoon.reflect.code.CtLocalVariable;
import spoon.reflect.declaration.CtExecutable;
import spoon.reflect.declaration.CtField;
import spoon.reflect.declaration.CtMethod;
import spoon.reflect.declaration.CtType;
import spoon.reflect.reference.CtExecutableReference;
import spoon.reflect.reference.CtFieldReference;
import spoon.reflect.reference.CtTypeReference;
import spoon.reflect.visitor.CtScanner;
import spoon.reflect.visitor.Query;
import spoon.reflect.visitor.filter.TypeFilter;

import java.util.List;

/**
 * Contains all methods to refactor code elements in the AST.
 */
public final class Refactoring {
	private Refactoring() { }

	/**
	 * Changes name of a type element.
	 *
	 * @param type
	 * 		Type in the AST.
	 * @param name
	 * 		New name of the element.
	 */
	public static void changeTypeName(final CtType type, String name) {

		// first we remove the type from the list of types
		// to be pretty-printed
		if (type.isTopLevel()) {
			type.getFactory().CompilationUnit().removeType(type);
		}

		final String typeQFN = type.getQualifiedName();
		final List> references = Query.getElements(type.getFactory(), new TypeFilter>(CtTypeReference.class) {
			@Override
			public boolean matches(CtTypeReference reference) {
				String refFQN = reference.getQualifiedName();
				return typeQFN.equals(refFQN);
			}
		});

		type.setSimpleName(name);
		for (CtTypeReference reference : references) {
			reference.setSimpleName(name);
		}

		// adding the new type
		if (type.isTopLevel()) {
			type.getFactory().CompilationUnit().addType(type);
		}

	}

	/**
	 * Changes name of a method, propagates the change in the executable references of the model.
	 */
	public static void changeMethodName(final CtMethod method, String newName) {

		final List> references = Query.getElements(method.getFactory(), new TypeFilter>(CtExecutableReference.class) {
			@Override
			public boolean matches(CtExecutableReference reference) {
				return reference.getDeclaration() == method;
			}
		});

		method.setSimpleName(newName);

		for (CtExecutableReference reference : references) {
			reference.setSimpleName(newName);
		}
	}

	/** See doc in {@link CtMethod#copyMethod()} */
	public static CtMethod copyMethod(final CtMethod method) {
		CtMethod clone = method.clone();
		StringBuilder tentativeTypeName = new StringBuilder(method.getSimpleName() + "Copy");
		CtType parent = method.getParent(CtType.class);
		while (!parent.getMethodsByName(tentativeTypeName.toString()).isEmpty()) {
			tentativeTypeName.append("X");
		}
		final String cloneMethodName = tentativeTypeName.toString();
		clone.setSimpleName(cloneMethodName);
		parent.addMethod(clone);
		new CtScanner() {
			@Override
			public  void visitCtExecutableReference(CtExecutableReference reference) {
				CtExecutable declaration = reference.getDeclaration();
				if (declaration == null) {
					return;
				}
				if (declaration == method) {
					reference.setSimpleName(cloneMethodName);
				}
				if (reference.getDeclaration() != clone) {
					throw new SpoonException("post condition broken " + reference);
				}
				super.visitCtExecutableReference(reference);

			}
		}.scan(clone);
		return clone;
	}


	/** See doc in {@link CtType#copyType()} */
	public static CtType copyType(final CtType type) {
		CtType clone = type.clone();
		StringBuilder tentativeTypeName = new StringBuilder(type.getSimpleName() + "Copy");
		while (type.getFactory().Type().get(type.getPackage().getQualifiedName() + "." + tentativeTypeName) != null) {
			tentativeTypeName.append("X");
		}
		final String cloneTypeName = tentativeTypeName.toString();
		clone.setSimpleName(cloneTypeName);
		type.getPackage().addType(clone);
		//fix cloned type name
		new CtScanner() {
			@Override
			public  void visitCtTypeReference(CtTypeReference reference) {
				if (reference.getDeclaration() == null) {
					return;
				}
				if (reference.getDeclaration() == type) {
					reference.setSimpleName(cloneTypeName);
				}
				super.visitCtTypeReference(reference);
			}

			@Override
			public  void visitCtExecutableReference(CtExecutableReference reference) {
				CtExecutable declaration = reference.getDeclaration();
				if (declaration == null) {
					return;
				}
				if (declaration.hasParent(type)) {
					reference.getDeclaringType().setSimpleName(cloneTypeName);
				}
				super.visitCtExecutableReference(reference);

			}

			@Override
			public  void visitCtFieldReference(CtFieldReference reference) {
				CtField declaration = reference.getDeclaration();
				if (declaration == null) {
					return;
				}
				if (declaration.hasParent(type)) {
					reference.getDeclaringType().setSimpleName(cloneTypeName);
				}
				super.visitCtFieldReference(reference);
			}

		}.scan(clone);
		//check that everything is OK
		new CtScanner() {
			@Override
			public  void visitCtTypeReference(CtTypeReference reference) {
				if (reference.getDeclaration() == null) {
					return;
				}
				if (reference.getDeclaration().getTopLevelType() != clone) {
					throw new SpoonException("post condition broken " + reference);
				}
				super.visitCtTypeReference(reference);
			}

			@Override
			public  void visitCtExecutableReference(CtExecutableReference reference) {
				CtExecutable declaration = reference.getDeclaration();
				if (declaration == null) {
					return;
				}
				if (!declaration.hasParent(clone)) {
					throw new SpoonException("post condition broken " + reference);
				}
				super.visitCtExecutableReference(reference);

			}

			@Override
			public  void visitCtFieldReference(CtFieldReference reference) {
				CtField declaration = reference.getDeclaration();
				if (declaration == null) {
					return;
				}
				if (!declaration.hasParent(clone)) {
					throw new SpoonException("post condition broken " + reference);
				}
				super.visitCtFieldReference(reference);
			}

		}.scan(clone);
		return clone;
	}

	/**
	 * Changes name of a {@link CtLocalVariable}.
	 *
	 * @param localVariable
	 * 		to be renamed {@link CtLocalVariable} in the AST.
	 * @param newName
	 * 		New name of the element.
	 * @throws RefactoringException when rename to newName would cause model inconsistency, like ambiguity, shadowing of other variables, etc.
	 */
	public static void changeLocalVariableName(CtLocalVariable localVariable, String newName) throws RefactoringException {
		new CtRenameLocalVariableRefactoring().setTarget(localVariable).setNewName(newName).refactor();
	}

	/**
	 * Removes all deprecated methods for all java files in the given path.
	 *
	 * Only use it if you have > jdk8. Older jdk versions may result in wrong results.
	 * Only removes deprecated methods, that are no longer invoked by any method or
	 * field. If a deprecated method is invoked by a deprecated method only, which
	 * can be removed, the deprecated method is removed too.
	 *
	 * Result is written to /$Path/$Package. For different output folder see
	 * {@link Refactoring#removeDeprecatedMethods(String, String)}.
	 *
	 * @param input Path to java files in folder.
	 */
	public static void removeDeprecatedMethods(String path) {
		new CtDeprecatedRefactoring().removeDeprecatedMethods(path);
	}

	/**
	 * Removes all deprecated methods for all java files in the given path.
	 *
	 * Only use it if you have > jdk8. Older jdk versions may result in wrong results.
	 * Only removes deprecated methods, that are no longer invoked by any method or
	 * field. If a deprecated method is invoked by a deprecated method only, which
	 * can be removed, the deprecated method is removed too.
	 *
	 * @param input      Path to java files in folder.
	 * @param resultPath Path for the refactored java files.
	 */
	public static void removeDeprecatedMethods(String input, String resultPath) {
		new CtDeprecatedRefactoring().removeDeprecatedMethods(input, resultPath);
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy