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

com.graphql_java_generator.client.GraphqlClientUtils Maven / Gradle / Ivy

There is a newer version: 1.18
Show newest version
/**
 * 
 */
package com.graphql_java_generator.client;

import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import org.springframework.stereotype.Component;

import com.graphql_java_generator.GraphqlUtils;
import com.graphql_java_generator.annotation.GraphQLEnumType;
import com.graphql_java_generator.annotation.GraphQLInputParameters;
import com.graphql_java_generator.annotation.GraphQLInterfaceType;
import com.graphql_java_generator.annotation.GraphQLNonScalar;
import com.graphql_java_generator.annotation.GraphQLObjectType;
import com.graphql_java_generator.annotation.GraphQLScalar;
import com.graphql_java_generator.annotation.GraphQLUnionType;
import com.graphql_java_generator.client.directive.Directive;
import com.graphql_java_generator.client.directive.DirectiveRegistry;
import com.graphql_java_generator.client.directive.DirectiveRegistryImpl;
import com.graphql_java_generator.client.request.InputParameter;
import com.graphql_java_generator.client.request.ObjectResponse;
import com.graphql_java_generator.customscalars.CustomScalarRegistryImpl;
import com.graphql_java_generator.exception.GraphQLRequestExecutionException;
import com.graphql_java_generator.exception.GraphQLRequestPreparationException;

import graphql.schema.GraphQLScalarType;

/**
 * @author etienne-sf
 */
@Component
public class GraphqlClientUtils {

	/** A singleton without Spring */
	public static GraphqlClientUtils graphqlClientUtils = new GraphqlClientUtils();

	/** The registry for all known GraphQL directives */
	DirectiveRegistry directiveRegistry = DirectiveRegistryImpl.directiveRegistry;

	Pattern graphqlNamePattern = Pattern.compile("^[_A-Za-z][_0-9A-Za-z]*$");

	GraphqlUtils graphqlUtils = new GraphqlUtils();

	/**
	 * maps for all scalers, when they are mandatory. The key is the type name. The value is the class to use in the
	 * java code
	 */
	List> scalars = new ArrayList<>();

	public GraphqlClientUtils() {
		// Add of all predefined scalars
		scalars.add(String.class);
		scalars.add(int.class);
		scalars.add(Integer.class);
		scalars.add(float.class);
		scalars.add(Float.class);
		scalars.add(boolean.class);
		scalars.add(Boolean.class);
	}

	/**
	 * Checks that the given GraphQL name is valid.
	 * 
	 * @param graphqlIdentifier
	 * @throws NullPointerException
	 *             If name is null
	 * @throws GraphQLRequestPreparationException
	 *             If the given graphqlIdentifier is not a valid identifier
	 */
	public void checkName(String graphqlIdentifier) throws GraphQLRequestPreparationException {
		if (graphqlIdentifier == null) {
			throw new NullPointerException("A GraphQL identifier may not be null");
		}
		Matcher m = graphqlNamePattern.matcher(graphqlIdentifier);
		if (!m.matches()) {
			throw new GraphQLRequestPreparationException("<" + graphqlIdentifier + "> is not a valid GraphQL name");
		}
	}

	/**
	 * This method checks whether the given field (as an attribute) of the given class is a GraphQL scalar, or not,
	 * depending on shouldBeScalar.
	 * 
	 * @param field
	 *            The field whose type should be (or not) a scalar
	 * @param shouldBeScalar
	 *            if true, this method checks that field's type is a scalar (if false, checks that it is not a scalar)
	 * @return Returns the Class indicated as the value for the graphqlType attribute of the GraphQLScalar or
	 *         GraphQLNonScalar annotation
	 * @throws GraphQLRequestPreparationException
	 */
	public Class checkIsScalar(java.lang.reflect.Field field, Boolean shouldBeScalar)
			throws GraphQLRequestPreparationException {
		// All types are generated in the same package. So, if the package for the field type is the same as the
		// package for our class, this means that the field is not a scalar
		// Note: we'll perhaps have to change that, depending on the way we manage scalar

		boolean isScalar = isScalar(field);

		if (shouldBeScalar != null) {
			if (shouldBeScalar & !isScalar) {
				throw new GraphQLRequestPreparationException("The field <" + field.getName() + "> of the GraphQL type <"
						+ field.getDeclaringClass().getName()
						+ "> is not a GraphQLScalar. At least one field must be defined for the server response.");
			}
			if (!shouldBeScalar & isScalar) {
				throw new GraphQLRequestPreparationException("The field <" + field.getName() + "> of the GraphQL type <"
						+ field.getDeclaringClass().getName()
						+ "> is not a GraphQLScalar. At least one field must be defined for the server response.");
			}
		}

		return getGraphQLType(field);
	}

	/**
	 * This method checks whether the given field (as a method: getter, query...) of the given class is a GraphQL
	 * scalar, or not, depending on shouldBeScalar.
	 * 
	 * @param fieldName
	 *            the name of the field represented by the given method.
	 * @param method
	 *            The method whose return should be (or not) a scalar. This method can be a setter, a getter (in which
	 *            case its name is different from the fieldName), or a query/mutation/subscription (in which case its
	 *            name is the fieldName)
	 * @param shouldBeScalar
	 *            if true, this method checks that method return type is a scalar (if false, checks that it is not a
	 *            scalar)
	 * @throws GraphQLRequestPreparationException
	 */
	public Class checkIsScalar(String fieldName, Method method, Boolean shouldBeScalar)
			throws GraphQLRequestPreparationException {
		// All types are generated in the same package. So, if the package for the field type is the same as the
		// package for our class, this means that the field is not a scalar
		// Note: we'll perhaps have to change that, depending on the way we manage scalar
		boolean isScalar = isScalar(method);

		if (method.getReturnType() == null) {
			throw new GraphQLRequestPreparationException("There is a method of name <" + fieldName
					+ "> in the GraphQL type <" + method.getDeclaringClass().getName()
					+ ">, but this method is a void method: it can't represent the <" + fieldName + "> GraphQL field");
		}

		if (shouldBeScalar != null) {
			if (shouldBeScalar && !isScalar) {
				throw new GraphQLRequestPreparationException(
						"The field <" + fieldName + "> (accessed through its getter: " + method.getName()
								+ ">) of the GraphQL type <" + method.getDeclaringClass().getName()
								+ "> should be a scalar. But is is actually not a GraphQLScalar");
			}
			if (!shouldBeScalar && isScalar) {
				throw new GraphQLRequestPreparationException(
						"The field <" + fieldName + "> (accessed through its getter: <" + method.getName()
								+ ">) of the GraphQL type <" + method.getDeclaringClass().getName()
								+ "> should not be a scalar. But is is actually a GraphQLScalar");
			}
		}

		return getGraphQLType(method);
	}

	/**
	 * Indicates whether the given class is a scalar or not
	 * 
	 * @param fieldOrMethod
	 * @return true if clazz is a scalar type
	 * @throws GraphQLRequestPreparationException
	 */
	public boolean isScalar(AccessibleObject fieldOrMethod) throws GraphQLRequestPreparationException {
		if (fieldOrMethod.getAnnotation(GraphQLScalar.class) != null
				|| fieldOrMethod.getAnnotation(GraphQLNonScalar.class) != null) {
			// Ok, at least on of GraphQLScalar and GraphQLNonScalar annotation is set.
			return fieldOrMethod.getAnnotation(GraphQLScalar.class) != null;
		} else {
			// No GraphQLScalar or GraphQLNonScalar annotation: let's throw an internal error.
			if (fieldOrMethod instanceof Field) {
				Field field = (Field) fieldOrMethod;
				throw new GraphQLRequestPreparationException("The field <" + field.getName() + "> of the class <"
						+ field.getDeclaringClass().getName()
						+ "> has none of the GraphQLCustomScalar, GraphQLScalar or GraphQLNonScalar annotation");
			} else {
				Method method = (Method) fieldOrMethod;
				throw new GraphQLRequestPreparationException("The method <" + method.getName() + "> of the class <"
						+ method.getDeclaringClass().getName()
						+ "> has none of the GraphQLCustomScalar, GraphQLScalar or GraphQLNonScalar annotation");
			}
		}
	}

	/**
	 * Returns the Class indicated as the value for the graphqlType attribute of the GraphQLScalar or GraphQLNonScalar
	 * annotation
	 * 
	 * @param fieldOrMethod
	 * @return
	 * @throws GraphQLRequestPreparationException
	 */
	public Class getGraphQLType(AccessibleObject fieldOrMethod) throws GraphQLRequestPreparationException {
		if (fieldOrMethod.getAnnotation(GraphQLScalar.class) != null) {
			return fieldOrMethod.getAnnotation(GraphQLScalar.class).javaClass();
		} else if (fieldOrMethod.getAnnotation(GraphQLNonScalar.class) != null) {
			return fieldOrMethod.getAnnotation(GraphQLNonScalar.class).javaClass();
		} else {
			// No GraphQLScalar or GraphQLNonScalar annotation: let's thrown an internal error.
			if (fieldOrMethod instanceof Field) {
				Field field = (Field) fieldOrMethod;
				throw new GraphQLRequestPreparationException("The field <" + field.getName() + "> of the class <"
						+ field.getDeclaringClass().getName()
						+ "> has none of the GraphQLCustomScalar, GraphQLScalar or GraphQLNonScalar annotation");
			} else {
				Method method = (Method) fieldOrMethod;
				throw new GraphQLRequestPreparationException("The method <" + method.getName() + "> of the class <"
						+ method.getDeclaringClass().getName()
						+ "> has none of the GraphQLCustomScalar, GraphQLScalar or GraphQLNonScalar annotation");
			}
		}
	}

	/**
	 * Check if the given field is owned by the class of this {@link ObjectResponse}. This method returns the class for
	 * this field.
	 * 
	 * @param name
	 *            The name of the field we want to check
	 * @param shouldBeScalar
	 *            if true: also checks that the field is a scalar (throws a GraphQLRequestPreparationException if not).
	 *            If false: also checks that the field is not a scalar (throws a GraphQLRequestPreparationException if
	 *            not). If null: no check whether the field is a scalar or not
	 * @param owningClass
	 *            The class in which will search for name as a GraphQL field
	 * @return the class of this field
	 * @throws NullPointerException
	 *             if name is null
	 * @throws GraphQLRequestPreparationException
	 *             if the check is KO
	 */
	public Class checkFieldOfGraphQLType(String name, Boolean shouldBeScalar, Class owningClass)
			throws GraphQLRequestPreparationException {

		// Let's be sure that the identifier is a valid GraphQL identifier (also checks that it's not null)
		checkName(name);

		// Let's check that this fieldName is either a method name or a field of the class for this ObjectResponse.
		Class fieldClass = null;

		Field field = graphqlUtils.getDeclaredField(owningClass, graphqlUtils.getJavaName(name), false);
		if (field != null) {
			// If we need to check that this field is (or is not) a scalar
			fieldClass = checkIsScalar(field, shouldBeScalar);
		}
		if (fieldClass == null && !owningClass.isInterface()) {
			// This class is a concrete class (not an interface). As the search field is not an attribute, the
			// owningClass should be a Query, a Mutation or a Subscription
			for (Method method : owningClass.getMethods()) {
				if (method.getName().equals(name)) {
					// If we need to check that this field is (or is not) a scalar
					fieldClass = checkIsScalar(name, method, shouldBeScalar);
					break;
				}
			}
		}
		if (fieldClass == null && owningClass.isInterface()) {
			// The class is an interface. So it's logical we didn't find this field as an attribute. Let's search for
			// the relevant setter
			String expectedMethodName = "get" + graphqlUtils.getPascalCase(name);
			for (Method method : owningClass.getDeclaredMethods()) {
				if (method.getName().equals(expectedMethodName)) {
					// If we need to check that this field is (or is not) a scalar
					fieldClass = checkIsScalar(name, method, shouldBeScalar);
					break;
				}
			}
		}

		if (fieldClass == null) {
			throw new GraphQLRequestPreparationException(
					"The GraphQL type <" + owningClass.getSimpleName() + "> has no field of name <" + name + ">");
		}

		return fieldClass;
	}

	/**
	 * This method retrieves the couple of name and values given in these parameters, stores them in a map where the key
	 * is the param name, and the value is the value of the {@link Map}.
	 * 
	 * @param paramsAndValues
	 *            A series of name and values : (paramName1, paramValue1, paramName2, paramValue2...). So there must be
	 *            an even number of items in this array. Empty arrays are allowed (that is no parameter name and
	 *            value).
* This series is sent by the developer's code, when it calls the request methods. * @return The map with paramName1, paramName2 (...) are the keys, and paramValue1, paramValue2 (...) are the * associated content. * @throws GraphQLRequestExecutionException * When a non-even number of parameters is sent to this method * @throws ClassCastException * When a parameter name is not a String */ public Map generatesBindVariableValuesMap(Object[] paramsAndValues) throws GraphQLRequestExecutionException, ClassCastException { Map map = new HashMap(); // If we get parameters and values, let's put them into the map if (paramsAndValues != null) { if (paramsAndValues.length % 2 != 0) { throw new GraphQLRequestExecutionException("An even number of parameters is expected, but " + paramsAndValues.length + " parameters where sent. This method expects a series of name and values : (paramName1, paramValue1, paramName2, paramValue2...)"); } for (int i = 0; i < paramsAndValues.length; i += 2) { map.put((String) paramsAndValues[i], paramsAndValues[i + 1]); } } return map; } /** * Retrieves the GraphQL type name (as defined in the GraphQL schema), from the GraphQL annotation added in the * generated code by the plugin. * * @param clazz * @return */ public String getGraphQLTypeNameFromClass(Class clazz) { // Object GraphQLObjectType graphQLObjectType = clazz.getAnnotation(GraphQLObjectType.class); if (graphQLObjectType != null) { return graphQLObjectType.value(); } // Interface GraphQLInterfaceType graphQLInterfaceType = clazz.getAnnotation(GraphQLInterfaceType.class); if (graphQLInterfaceType != null) { return graphQLInterfaceType.value(); } // Union GraphQLUnionType graphQLUnionType = clazz.getAnnotation(GraphQLUnionType.class); if (graphQLUnionType != null) { return graphQLUnionType.value(); } // Enum GraphQLEnumType graphQLEnumType = clazz.getAnnotation(GraphQLEnumType.class); if (graphQLEnumType != null) { return graphQLEnumType.value(); } throw new RuntimeException("Could not find the GraphQL type for the class " + clazz.getName()); } /** * This method retrieves the {@link GraphQLScalarType} for a custom scalar field or method. {@link GraphQLScalar} is * used in generated InputType and response type POJOs and {@link GraphQLCustomScalar} is used in some legacy * generated response type POJOs. * * @param fieldOrMethod * The field or method of the generated POJO class * @return the {@link GraphQLScalarType} */ public GraphQLScalarType getGraphQLCustomScalarType(AccessibleObject fieldOrMethod) { String graphQLTypeName; if (fieldOrMethod.getAnnotation(GraphQLScalar.class) != null) { graphQLTypeName = fieldOrMethod.getAnnotation(GraphQLScalar.class).graphQLTypeSimpleName(); } else { graphQLTypeName = null; } if (graphQLTypeName != null) { return CustomScalarRegistryImpl.customScalarRegistry.getGraphQLScalarType(graphQLTypeName); } else { return null; } } /** * Retrieves the {@link GraphQLScalarType} from this input parameter, if this parameter is a Custom Scalar * * @param directive * If not null, then we're looking for an argument of a GraphQL directive. Oherwise, it's a field * argument, and the owningClass and fieldName parameters will be used. * @param owningClass * The class that contains this field * @param fieldName * The field name * @param parameterName * The parameter name, which must be the name for an input parameter for this field in the GraphQL schema * @return The {@link GraphQLScalarType} if this type is a scalar (out of enums). And null otherwise. * @throws GraphQLRequestPreparationException */ public GraphQLScalarType getGraphQLType(Directive directive, Class owningClass, String fieldName, String parameterName) throws GraphQLRequestPreparationException { if (directive != null) { // Let's find the definition for this directive Directive dirDef = directiveRegistry.getDirective(directive.getName()); if (dirDef == null) { throw new GraphQLRequestPreparationException( "Could not find directive definition for the directive '" + directive.getName() + "'"); } // Let's find the GraphQL type of this argument for (InputParameter param : dirDef.getArguments()) { if (param.getName().equals(parameterName)) { return param.getGraphQLScalarType(); } } // for throw new GraphQLRequestPreparationException("The parameter of name '" + parameterName + "' has not been found for the directive '" + directive.getName() + "'"); } else { GraphQLInputParameters inputParams; if (owningClass.isInterface()) { Method method; try { method = owningClass .getMethod("get" + graphqlUtils.getPascalCase(graphqlUtils.getJavaName(fieldName))); inputParams = method.getAnnotation(GraphQLInputParameters.class); } catch (NoSuchMethodException | SecurityException e) { throw new GraphQLRequestPreparationException("Error while looking for the the getter for <" + fieldName + "> in the interface '" + owningClass.getName() + "'", e); } } else { try { Field field = owningClass.getDeclaredField(graphqlUtils.getJavaName(fieldName)); inputParams = field.getAnnotation(GraphQLInputParameters.class); } catch (NoSuchFieldException | SecurityException e) { // We may be in the XxxResponse class. ITs has no field, and all the relevant fields are in its // superclass. Let's recurse once. try { Field field = owningClass.getSuperclass().getDeclaredField(graphqlUtils.getJavaName(fieldName)); inputParams = field.getAnnotation(GraphQLInputParameters.class); } catch (NoSuchFieldException | SecurityException e2) { // We may be in the XxxResponse class. ITs has no field, and all the relevant fields are in its // superclass. Let's recurse once. throw new GraphQLRequestPreparationException("Error while looking for the the field <" + fieldName + "> in the class '" + owningClass.getName() + "', not in its superclass: " + owningClass.getSuperclass().getName(), e); } } } if (inputParams == null) throw new GraphQLRequestPreparationException("The field <" + fieldName + "> of the class '" + owningClass.getName() + "' has no input parameters. Error while looking for its '" + parameterName + "' input parameter"); for (int i = 0; i < inputParams.names().length; i += 1) { if (inputParams.names()[i].equals(parameterName)) { // We've found the expected parameter return getGraphQLTypeFromName(inputParams.types()[i]); } } throw new GraphQLRequestPreparationException( "The parameter of name <" + parameterName + "> has not been found for the field <" + fieldName + "> of the class '" + owningClass.getName() + "'"); } } /** * Returns the GraphQL type for this object * * @param typeName * @return The GraphQL type. Or null if not found (enum, object, input type, interface, union) * @throws GraphQLRequestPreparationException */ public GraphQLScalarType getGraphQLTypeFromName(String typeName) throws GraphQLRequestPreparationException { // Is it a known type ? if (typeName.equals("String")) { return graphql.Scalars.GraphQLString; } else if (typeName.equals("Boolean")) { return graphql.Scalars.GraphQLBoolean; } else if (typeName.equals("Float")) { return graphql.Scalars.GraphQLFloat; } else if (typeName.equals("Int")) { return graphql.Scalars.GraphQLInt; } else if (typeName.equals("ID")) { return graphql.Scalars.GraphQLID; } return CustomScalarRegistryImpl.customScalarRegistry.getGraphQLScalarType(typeName); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy