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

lombok.core.handlers.BuilderAndExtensionHandler Maven / Gradle / Ivy

The newest version!
/*
 * Copyright © 2011-2012 Philipp Eichhorn
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package lombok.core.handlers;

import static lombok.ast.AST.*;
import static lombok.ast.IMethod.ArgumentStyle.INCLUDE_ANNOTATIONS;
import static lombok.ast.Wildcard.Bound.EXTENDS;
import static lombok.core.TransformationsUtil.NON_NULL_PATTERN;
import static lombok.core.util.Names.*;

import java.util.*;

import lombok.*;
import lombok.ast.*;
import lombok.core.util.Is;
import lombok.core.util.Names;

public class BuilderAndExtensionHandler, METHOD_TYPE extends IMethod, FIELD_TYPE extends IField> {
	public static final String OPTIONAL_DEF = "OptionalDef";
	public static final String BUILDER = "$Builder";

	public void handleBuilder(final TYPE_TYPE type, final Builder builder) {
		final BuilderData builderData = new BuilderData(type, builder).collect();
		final List interfaceTypes = new ArrayList(builderData.getRequiredFieldDefTypes());
		interfaceTypes.add(Type(OPTIONAL_DEF));
		for (TypeRef interfaceType : interfaceTypes) interfaceType.withTypeArguments(type.typeArguments());
		final List> builderMethods = new ArrayList>();

		createConstructor(builderData);
		createInitializeBuilderMethod(builderData);
		createRequiredFieldInterfaces(builderData, builderMethods);
		createOptionalFieldInterface(builderData, builderMethods);
		createBuilder(builderData, interfaceTypes, builderMethods);
	}

	public void handleExtension(final TYPE_TYPE type, final METHOD_TYPE method, final IParameterValidator validation,
			final IParameterSanitizer sanitizer, final Builder builder, final Builder.Extension extension) {
		TYPE_TYPE builderType = type. memberType(BUILDER);
		final BuilderData builderData = new BuilderData(type, builder).collect();

		final ExtensionType extensionType = getExtensionType(method, builderData, extension.fields());
		if (extensionType == ExtensionType.NONE) return;

		TYPE_TYPE interfaceType;
		if (extensionType  == ExtensionType.REQUIRED) {
			interfaceType = type. memberType(builderData.getRequiredFieldDefTypeNames().get(0));
		} else {
			interfaceType = type. memberType(OPTIONAL_DEF);
		}
		builderType.editor().injectMethod(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), method.name()).posHint(method.get()).makePublic().implementing().withArguments(method.arguments(INCLUDE_ANNOTATIONS)) //
				.withStatements(validation.validateParameterOf(method)) //
				.withStatements(sanitizer.sanitizeParameterOf(method)) //
				.withStatements(method.statements()) //
				.withStatement(Return(This())));
		interfaceType.editor().injectMethod(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), method.name()).makePublic().withNoBody().withArguments(method.arguments(INCLUDE_ANNOTATIONS)));
		type.editor().removeMethod(method);
	}

	private ExtensionType getExtensionType(final METHOD_TYPE method, final BuilderData builderData, final String[] fields) {
		if (method.isConstructor() || (method.accessLevel() != AccessLevel.PRIVATE) || !method.returns("void")) {
			method.node().addWarning("@Builder.Extension: The method '" + method.name() + "' is not a valid extension and was ignored.");
			return ExtensionType.NONE;
		}

		final String[] extensionFieldNames = Is.notEmpty(fields) ? fields : extensionFieldNames(method, builderData);
		List allFieldNames = builderData.getAllFieldNames();
		for (String potentialFieldName : extensionFieldNames) {
			if (!allFieldNames.contains(Names.decapitalize(potentialFieldName))) {
				method.node().addWarning("@Builder.Extension: The method '" + method.name() + "' is not a valid extension and was ignored.");
				return ExtensionType.NONE;
			}
		}

		List requiredFieldNames = builderData.getRequiredFieldNames();
		Set uninitializedRequiredFieldNames = new HashSet();
		for (FIELD_TYPE field : builderData.getAllFields()) {
			if (requiredFieldNames.contains(field.name()) && !field.isInitialized()) {
				uninitializedRequiredFieldNames.add(field.name());
			}
		}

		boolean containsRequiredFields = false;
		for (String potentialFieldName : extensionFieldNames) {
			containsRequiredFields |= uninitializedRequiredFieldNames.remove(Names.decapitalize(potentialFieldName));
		}

		if (containsRequiredFields) {
			if (uninitializedRequiredFieldNames.isEmpty()) {
				return ExtensionType.REQUIRED;
			} else {
				method.node().addWarning("@Builder.Extension: The method '" + method.name() + "' is not a valid extension and was ignored.");
				return ExtensionType.NONE;
			}
		}
		return ExtensionType.OPTIONAL;
	}

	private String[] extensionFieldNames(final METHOD_TYPE method, final BuilderData builderData) {
		final String prefix = builderData.getPrefix();
		String methodName = method.name();
		if (methodName.startsWith(prefix)) {
			methodName = methodName.substring(prefix.length());
		}
		return methodName.split("And");
	}

	private void createConstructor(final BuilderData builderData) {
		TYPE_TYPE type = builderData.getType();

		if (hasCustomConstructor(type)) return;

		ConstructorDecl constructorDecl = ConstructorDecl(type.name()).makePrivate().withArgument(Arg(Type(BUILDER).withTypeArguments(type.typeArguments()), "builder").makeFinal()).withImplicitSuper();
		for (final FIELD_TYPE field : builderData.getAllFields()) {
			if (field.isFinal() && field.isInitialized()) {
				if (isCollection(field)) {
					constructorDecl.withStatement(Call(Field(field.name()), "addAll").withArgument(Field(Name("builder"), field.name())));
				} else if (isMap(field)) {
					constructorDecl.withStatement(Call(Field(field.name()), "putAll").withArgument(Field(Name("builder"), field.name())));
				}
			} else {
				constructorDecl.withStatement(Assign(Field(field.name()), Field(Name("builder"), field.name())));
			}
		}
		type.editor().injectConstructor(constructorDecl);
	}

	private boolean hasCustomConstructor(final IType type) {
		for (final METHOD_TYPE method : type.methods()) {
			if (!method.isConstructor()) continue;
			final List arguments = method.arguments();
			if (arguments.size() != 1) continue;
			final Argument argument = arguments.get(0);
			final String argumentTypeName = argument.getType().toString();
			if (argumentTypeName.endsWith("Builder")) {
				method.editor().replaceArguments(Arg(Type(BUILDER).withTypeArguments(type.typeArguments()), argument.getName()).makeFinal());
				return true;
			}
		}
		return false;
	}

	private void createInitializeBuilderMethod(final BuilderData builderData) {
		final TYPE_TYPE type = builderData.getType();
		final TypeRef fieldDefType = builderData.getRequiredFields().isEmpty() ? Type(OPTIONAL_DEF) : builderData.getRequiredFieldDefTypes().get(0);
		type.editor().injectMethod(MethodDecl(fieldDefType, decapitalize(type.name())).makeStatic().withAccessLevel(builderData.getLevel()).withTypeParameters(type.typeParameters()) //
				.withStatement(Return(New(Type(BUILDER).withTypeArguments(type.typeArguments())))));
	}

	private void createRequiredFieldInterfaces(final BuilderData builderData, final List> builderMethods) {
		List fields = builderData.getRequiredFields();
		if (!fields.isEmpty()) {
			TYPE_TYPE type = builderData.getType();
			List names = builderData.getRequiredFieldDefTypeNames();
			FIELD_TYPE field = fields.get(0);
			String name = names.get(0);
			for (int i = 1, iend = fields.size(); i < iend; i++) {
				List> interfaceMethods = new ArrayList>();
				createFluentSetter(builderData, names.get(i), field, interfaceMethods, builderMethods);

				type.editor().injectType(InterfaceDecl(name).makePublic().makeStatic().withTypeParameters(type.typeParameters()).withMethods(interfaceMethods));
				field = fields.get(i);
				name = names.get(i);
			}
			List> interfaceMethods = new ArrayList>();
			createFluentSetter(builderData, OPTIONAL_DEF, field, interfaceMethods, builderMethods);

			type.editor().injectType(InterfaceDecl(name).makePublic().makeStatic().withTypeParameters(type.typeParameters()).withMethods(interfaceMethods));
		}
	}

	private void createOptionalFieldInterface(final BuilderData builderData, final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		List> interfaceMethods = new ArrayList>();
		for (FIELD_TYPE field : builderData.getOptionalFields()) {
			if (isInitializedMapOrCollection(field)) {
				if (builderData.isGenerateConvenientMethodsEnabled()) {
					if (isCollection(field)) {
						createCollectionMethods(builderData, field, interfaceMethods, builderMethods);
					} else if (isMap(field)) {
						createMapMethods(builderData, field, interfaceMethods, builderMethods);
					}
				}
			} else {
				createFluentSetter(builderData, OPTIONAL_DEF, field, interfaceMethods, builderMethods);
			}
		}

		createBuildMethod(builderData, type.name(), interfaceMethods, builderMethods);

		if (builderData.isAllowReset()) {
			createResetMethod(builderData, interfaceMethods, builderMethods);
		}

		for (String callMethod : builderData.getCallMethods()) {
			createMethodCall(builderData, callMethod, interfaceMethods, builderMethods);
		}

		type.editor().injectType(InterfaceDecl(OPTIONAL_DEF).makePublic().makeStatic().withTypeParameters(type.typeParameters()).withMethods(interfaceMethods));
	}

	private void createFluentSetter(final BuilderData builderData, final String typeName, final FIELD_TYPE field,
			final List> interfaceMethods, final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		String methodName = camelCase(builderData.getPrefix(), field.name());
		final Argument arg0 = Arg(field.type(), field.name()).makeFinal();
		builderMethods.add(MethodDecl(Type(typeName).withTypeArguments(type.typeArguments()), methodName).makePublic().implementing().withArgument(arg0) //
				.withStatement(Assign(Field(field.name()), Name(field.name()))) //
				.withStatement(Return(This())));
		interfaceMethods.add(MethodDecl(Type(typeName).withTypeArguments(type.typeArguments()), methodName).makePublic().withNoBody().withArgument(arg0));
	}

	private void createCollectionMethods(final BuilderData builderData, final FIELD_TYPE field,
			final List> interfaceMethods, final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		TypeRef elementType = Type(Object.class);
		TypeRef collectionType = Type(Collection.class);
		List typeArguments = field.typeArguments();
		if (typeArguments.size() == 1) {
			elementType = typeArguments.get(0);
			collectionType.withTypeArgument(Wildcard(EXTENDS, elementType));
		}

		{ // add
			String addMethodName = singular(camelCase(builderData.getPrefix(), field.name()));
			final Argument arg0 = Arg(elementType, "arg0").makeFinal();
			builderMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), addMethodName).makePublic().implementing().withArgument(arg0) //
					.withStatement(Call(Field(field.name()), "add").withArgument(Name("arg0"))) //
					.withStatement(Return(This())));
			interfaceMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), addMethodName).makePublic().withNoBody().withArgument(arg0));
		}
		{ // addAll
			String addAllMethodName = camelCase(builderData.getPrefix(), field.name());
			final Argument arg0 = Arg(collectionType, "arg0").makeFinal();
			builderMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), addAllMethodName).makePublic().implementing().withArgument(arg0) //
					.withStatement(Call(Field(field.name()), "addAll").withArgument(Name("arg0"))) //
					.withStatement(Return(This())));
			interfaceMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), addAllMethodName).makePublic().withNoBody().withArgument(arg0));
		}
	}

	private void createMapMethods(final BuilderData builderData, final FIELD_TYPE field, final List> interfaceMethods,
			final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		TypeRef keyType = Type(Object.class);
		TypeRef valueType = Type(Object.class);
		TypeRef mapType = Type(Map.class);
		List typeArguments = field.typeArguments();
		if (typeArguments.size() == 2) {
			keyType = typeArguments.get(0);
			valueType = typeArguments.get(1);
			mapType.withTypeArgument(Wildcard(EXTENDS, keyType)) //
					.withTypeArgument(Wildcard(EXTENDS, valueType));
		}

		{ // put
			final String putMethodName = singular(camelCase(builderData.getPrefix(), field.name()));
			final Argument arg0 = Arg(keyType, "arg0").makeFinal();
			final Argument arg1 = Arg(valueType, "arg1").makeFinal();
			builderMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), putMethodName).makePublic().implementing().withArgument(arg0).withArgument(arg1) //
					.withStatement(Call(Field(field.name()), "put").withArgument(Name("arg0")).withArgument(Name("arg1")))
					.withStatement(Return(This())));
			interfaceMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), putMethodName).makePublic().withNoBody().withArgument(arg0).withArgument(arg1));
		}
		{ // putAll
			String putAllMethodName = camelCase(builderData.getPrefix(), field.name());
			final Argument arg0 = Arg(mapType, "arg0").makeFinal();
			builderMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), putAllMethodName).makePublic().implementing().withArgument(arg0) //
					.withStatement(Call(Field(field.name()), "putAll").withArgument(Name("arg0"))) //
					.withStatement(Return(This())));
			interfaceMethods.add(MethodDecl(Type(OPTIONAL_DEF).withTypeArguments(type.typeArguments()), putAllMethodName).makePublic().withNoBody().withArgument(arg0));
		}
	}

	private void createBuildMethod(final BuilderData builderData, final String typeName, final List> interfaceMethods,
			final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		builderMethods.add(MethodDecl(Type(typeName).withTypeArguments(type.typeArguments()), "build").makePublic().implementing() //
				.withStatement(Return(New(Type(typeName).withTypeArguments(type.typeArguments())).withArgument(This()))));
		interfaceMethods.add(MethodDecl(Type(typeName).withTypeArguments(type.typeArguments()), "build").makePublic().withNoBody());
	}

	private void createResetMethod(final BuilderData builderData, final List> interfaceMethods,
			final List> builderMethods) {
		final TypeRef fieldDefType = builderData.getRequiredFields().isEmpty() ? Type(OPTIONAL_DEF) : builderData.getRequiredFieldDefTypes().get(0);
		MethodDecl methodDecl = MethodDecl(fieldDefType, "reset").makePublic().implementing();
		for (final FIELD_TYPE field : builderData.getAllFields()) {
			if (field.isInitialized()) {
				String fieldDefaultMethodName = "$" + field.name() + "Default";
				methodDecl.withStatement(Assign(Field(field.name()), Call(fieldDefaultMethodName)));
			} else {
				methodDecl.withStatement(Assign(Field(field.name()), DefaultValue(field.type())));
			}
		}
		builderMethods.add(methodDecl.withStatement(Return(This())));
		interfaceMethods.add(MethodDecl(fieldDefType, "reset").makePublic().withNoBody());
	}

	private void createMethodCall(final BuilderData builderData, final String methodName, final List> interfaceMethods,
			final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();

		TypeRef returnType = Type("void");
		boolean returnsVoid = true;
		List thrownExceptions = new ArrayList();
		if ("toString".equals(methodName)) {
			returnType = Type(String.class);
			returnsVoid = false;
		} else {
			for (METHOD_TYPE method : type.methods()) {
				if (methodName.equals(method.name()) && !method.hasArguments()) {
					returnType = method.returns();
					returnsVoid = method.returns("void");
					thrownExceptions.addAll(method.thrownExceptions());
					break;
				}
			}
		}

		Call call = Call(Call("build"), methodName);
		if (returnsVoid) {
			builderMethods.add(MethodDecl(returnType, methodName).makePublic().implementing().withThrownExceptions(thrownExceptions) //
					.withStatement(call));
		} else {
			builderMethods.add(MethodDecl(returnType, methodName).makePublic().implementing().withThrownExceptions(thrownExceptions) //
					.withStatement(Return(call)));
		}
		interfaceMethods.add(MethodDecl(returnType, methodName).makePublic().withNoBody().withThrownExceptions(thrownExceptions));
	}

	private void createBuilder(final BuilderData builderData, final List interfaceTypes,
			final List> builderMethods) {
		TYPE_TYPE type = builderData.getType();
		List builderFields = new ArrayList();
		List> builderFieldDefaultMethods = new ArrayList>();
		for (FIELD_TYPE field : builderData.getAllFields()) {
			FieldDecl builderField = FieldDecl(field.type(), field.name()).makePrivate();
			if (field.isInitialized()) {
				String fieldDefaultMethodName = "$" + field.name() + "Default";
				builderFieldDefaultMethods.add(MethodDecl(field.type(), fieldDefaultMethodName).makeStatic().withTypeParameters(type.typeParameters()) //
						.withStatement(Return(field.initialization())));
				builderField.withInitialization(Call(fieldDefaultMethodName));
				field.editor().replaceInitialization(Call(Name(BUILDER), fieldDefaultMethodName));
			}
			builderFields.add(builderField);
		}
		type.editor().injectType(ClassDecl(BUILDER).withTypeParameters(type.typeParameters()).makePrivate().makeStatic().implementing(interfaceTypes) //
				.withFields(builderFields) //
				.withMethods(builderFieldDefaultMethods) //
				.withMethods(builderMethods) //
				.withMethod(ConstructorDecl(BUILDER).makePrivate().withImplicitSuper()));
	}

	private static > boolean isInitializedMapOrCollection(final FIELD_TYPE field) {
		return (isMap(field) || isCollection(field)) && field.isInitialized();
	}

	private static > boolean isCollection(final FIELD_TYPE field) {
		return (field.isOfType("Collection") || field.isOfType("List") || field.isOfType("Set"));
	}

	private static > boolean isMap(final FIELD_TYPE field) {
		return field.isOfType("Map");
	}

	@Getter
	private static class BuilderData, METHOD_TYPE extends IMethod, FIELD_TYPE extends IField> {
		private final List requiredFields = new ArrayList();
		private final List optionalFields = new ArrayList();
		private final List requiredFieldDefTypes = new ArrayList();
		private final List requiredFieldNames = new ArrayList();
		private final List optionalFieldNames = new ArrayList();
		private final List requiredFieldDefTypeNames = new ArrayList();;
		private final TYPE_TYPE type;
		private final String prefix;
		private final List callMethods;
		private final boolean generateConvenientMethodsEnabled;
		private final boolean allowReset;
		private final AccessLevel level;
		private final Set excludes;

		private BuilderData(final TYPE_TYPE type, final Builder builder) {
			this.type = type;
			excludes = new HashSet(Arrays.asList(builder.exclude()));
			generateConvenientMethodsEnabled = builder.convenientMethods();
			prefix = builder.prefix();
			callMethods = Arrays.asList(builder.callMethods());
			level = builder.value();
			allowReset = builder.allowReset();
		}

		public BuilderData collect() {
			for (FIELD_TYPE field : type.fields()) {
				if (field.isStatic()) continue;
				String fieldName = field.name();
				if (excludes.contains(fieldName)) continue;
				if ((!field.isInitialized()) && (field.isFinal() || !field.annotations(NON_NULL_PATTERN).isEmpty())) {
					requiredFields.add(field);
					requiredFieldNames.add(fieldName);
					String typeName = capitalize(camelCase(fieldName, "def"));
					requiredFieldDefTypeNames.add(typeName);
					requiredFieldDefTypes.add(Type(typeName));
				} else if ((generateConvenientMethodsEnabled && isInitializedMapOrCollection(field)) || !field.isFinal()) {
					optionalFields.add(field);
					optionalFieldNames.add(fieldName);
				}
			}
			return this;
		}

		public List getAllFields() {
			List allFields = new ArrayList(getRequiredFields());
			allFields.addAll(getOptionalFields());
			return allFields;
		}

		public List getAllFieldNames() {
			List allFieldNames = new ArrayList(getRequiredFieldNames());
			allFieldNames.addAll(getOptionalFieldNames());
			return allFieldNames;
		}
	}

	private enum ExtensionType {
		NONE,
		REQUIRED,
		OPTIONAL;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy