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

io.helidon.microprofile.graphql.server.SchemaFieldDefinition Maven / Gradle / Ivy

/*
 * Copyright (c) 2020, 2023 Oracle and/or its affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package io.helidon.microprofile.graphql.server;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

import graphql.schema.DataFetcher;

/**
 * The representation of a GraphQL Field Definition.
 */
class SchemaFieldDefinition extends AbstractDescriptiveElement implements ElementGenerator {
    /**
     * Name of the field definition.
     */
    private final String name;

    /**
     * Return type.
     */
    private String returnType;

    /**
     * Indicates if the return type is an array type such as a native array([]) or a List, Collection, etc.
     */
    private final boolean isArrayReturnType;

    /**
     * The number of array levels if return type is an array.
     */
    private final int arrayLevels;

    /**
     * Indicates if the return type is mandatory.
     */
    private final boolean isReturnTypeMandatory;

    /**
     * List of arguments.
     */
    private final List listSchemaArguments;

    /**
     * If the return type is an array then indicates if the value in the
     * array is mandatory.
     */
    private boolean isArrayReturnTypeMandatory;

    /**
     * {@link DataFetcher} to override default behaviour of field.
     */
    private DataFetcher dataFetcher;

    /**
     * Original type before it was converted to a GraphQL representation.
     */
    private Class originalType;

    /**
     * Original array inner type if it is array type.
     */
    private Class originalArrayType;

    /**
     * Defines the format for a number or date.
     */
    private String[] format;

    /**
     * The default value for this field definition. Only valid for field definitions of an input type.
     */
    private Object defaultValue;

    /**
     * Indicates if the field has a default format applied (such as default date format) rather than a specific format supplied.
     */
    private boolean defaultFormatApplied;

    /**
     * Indicates if the format is of type Jsonb.
     */
    private boolean isJsonbFormat;

    /**
     * Indicates if the property name is of type Jsonb.
     */
    private boolean isJsonbProperty;

    /**
     * Construct a {@link SchemaFieldDefinition}.
     *
     * @param builder the {@link Builder} to construct from
     */
    private SchemaFieldDefinition(Builder builder) {
        this.name = builder.name;
        this.returnType = builder.returnType;
        this.isArrayReturnType = builder.isArrayReturnType;
        this.arrayLevels = builder.arrayLevels;
        this.isReturnTypeMandatory = builder.isReturnTypeMandatory;
        this.listSchemaArguments = builder.listSchemaArguments;
        this.isArrayReturnTypeMandatory = builder.isArrayReturnTypeMandatory;
        this.dataFetcher = builder.dataFetcher;
        this.originalType = builder.originalType;
        this.originalArrayType = builder.originalArrayType;
        this.format = builder.format;
        this.defaultValue = builder.defaultValue;
        this.defaultFormatApplied = builder.defaultFormatApplied;
        this.isJsonbFormat = builder.isJsonbFormat;
        this.isJsonbProperty = builder.isJsonbProperty;
        description(builder.description);
    }

    /**
     * Fluent API builder to create {@link SchemaFieldDefinition}.
     *
     * @return new builder instance
     */
    public static Builder builder() {
        return new Builder();
    }

    @Override
    public String getSchemaAsString() {
        StringBuilder sb = new StringBuilder(getSchemaElementDescription(format()))
                .append(name());

        // determine if there are any arguments that are DataFetcherEnvironment as they should
        // not be included as standard types
        boolean hasSchemaArguments = listSchemaArguments.stream().anyMatch(a -> !a.isDataFetchingEnvironment());

        if (hasSchemaArguments) {
            sb.append(OPEN_PARENTHESES)
                    .append(NEWLINE)
                    .append(listSchemaArguments.stream()
                                    .filter(a -> !a.isDataFetchingEnvironment())
                                    .map(SchemaArgument::getSchemaAsString)
                                    .collect(Collectors.joining(COMMA_SPACE + NEWLINE)));
            sb.append(NEWLINE).append(CLOSE_PARENTHESES);
        }

        sb.append(COLON);

        if (isArrayReturnType()) {
            int count = arrayLevels();
            sb.append(SPACER).append(repeat(count, OPEN_SQUARE))
                    .append(returnType())
                    .append(isArrayReturnTypeMandatory() ? MANDATORY : NOTHING)
                    .append(repeat(count, CLOSE_SQUARE));
        } else {
            sb.append(SPACER).append(returnType());
        }

        if (isReturnTypeMandatory()) {
            sb.append(MANDATORY);
        }

        if (defaultValue != null) {
            sb.append(generateDefaultValue(defaultValue, returnType()));
        }

        return sb.toString();
    }

    /**
     * Return the name for the field definition.
     *
     * @return the name for the field definition
     */
    public String name() {
        return name;
    }

    /**
     * Return the {@link List} of {@link SchemaArgument}s.
     *
     * @return the {@link List} of {@link SchemaArgument}s
     */
    public List arguments() {
        return listSchemaArguments;
    }

    /**
     * Return the return type.
     *
     * @return the return type
     */
    public String returnType() {
        return returnType;
    }

    /**
     * Indicates if the return type is an array type.
     *
     * @return if the return type is an array type
     */
    public boolean isArrayReturnType() {
        return isArrayReturnType;
    }

    /**
     * Indicates if the return type is mandatory.
     *
     * @return if the return type is mandatory
     */
    public boolean isReturnTypeMandatory() {
        return isReturnTypeMandatory;
    }

    /**
     * Return the {@link DataFetcher} for this {@link SchemaFieldDefinition}.
     *
     * @return he {@link DataFetcher} for this {@link SchemaFieldDefinition}
     */
    public DataFetcher dataFetcher() {
        return dataFetcher;
    }

    /**
     * Set the return type.
     *
     * @param sReturnType the return type
     */
    public void returnType(String sReturnType) {
        returnType = sReturnType;
    }

    /**
     * Set the {@link DataFetcher} which will override the default {@link DataFetcher} for the field.
     *
     * @param dataFetcher the {@link DataFetcher}
     */
    public void dataFetcher(DataFetcher dataFetcher) {
        this.dataFetcher = dataFetcher;
    }

    /**
     * Add a {@link SchemaArgument} to this {@link SchemaFieldDefinition}.
     *
     * @param schemaArgument {@link SchemaArgument} to add
     */
    public void addArgument(SchemaArgument schemaArgument) {
        listSchemaArguments.add(schemaArgument);
    }

    /**
     * Return the number of array levels if return type is an array.
     *
     * @return the number of array levels if return type is an array
     */
    public int arrayLevels() {
        return arrayLevels;
    }

    /**
     * Return the format for a number or date.
     *
     * @return the format for a number or date
     */
    public String[] format() {
        if (format == null) {
            return null;
        }
        String[] copy = new String[format.length];
        System.arraycopy(format, 0, copy, 0, copy.length);
        return copy;
    }

    /**
     * Set the format for a number or date.
     *
     * @param format the format for a number or date
     */
    public void format(String[] format) {
        if (format == null) {
            this.format = null;
        } else {
            this.format = new String[format.length];
            System.arraycopy(format, 0, this.format, 0, this.format.length);
        }
    }

    /**
     * Return the default value for this field definition.
     *
     * @return the default value for this field definition
     */
    public Object defaultValue() {
        return defaultValue;
    }

    /**
     * Set the default value for this field definition.
     *
     * @param defaultValue the default value for this field definition
     */
    public void defaultValue(Object defaultValue) {
        this.defaultValue = defaultValue;
    }

    /**
     * Sets the original return type.
     *
     * @param originalType the original return type
     */
    public void originalType(Class originalType) {
        this.originalType = originalType;
    }

    /**
     * Retrieve the original return type.
     *
     * @return the original return type
     */
    public Class originalType() {
        return originalType;
    }

    /**
     * Sets the original array type.
     *
     * @param originalArrayType the original array type
     */
    public void originalArrayType(Class originalArrayType) {
        this.originalArrayType = originalArrayType;
    }

    /**
     * Return the original array type.
     *
     * @return the original array type
     */
    public Class originalArrayType() {
        return originalArrayType;
    }

    /**
     * Return if the array return type is mandatory.
     * @return if the array return type is mandatory
     */
    public boolean isArrayReturnTypeMandatory() {
        return isArrayReturnTypeMandatory;
    }

    /**
     * Sets if the array return type is mandatory.
     * @param arrayReturnTypeMandatory if the array return type is mandatory
     */
    public void arrayReturnTypeMandatory(boolean arrayReturnTypeMandatory) {
        isArrayReturnTypeMandatory = arrayReturnTypeMandatory;
    }

    /**
     * Set if the field has a default format applied.
     * @param defaultFormatApplied if the field has a default format applied
     */
    public void defaultFormatApplied(boolean defaultFormatApplied) {
        this.defaultFormatApplied = defaultFormatApplied;
    }

    /**
     * Return if the field has a default format applied.
     * @return if the field has a default format applied
     */
    public boolean isDefaultFormatApplied() {
        return defaultFormatApplied;
    }

    /**
     * Set if the format is of type Jsonb.
     * @param isJsonbFormat if the format is of type Jsonb
     */
    public void jsonbFormat(boolean isJsonbFormat) {
        this.isJsonbFormat = isJsonbFormat;
    }

    /**
     * Return true if the format is of type Jsonb.
     * @return true if the format is of type Jsonb
     */
    public boolean isJsonbFormat() {
        return isJsonbFormat;
    }

    /**
     * Sets if the property has a JsonbProperty annotation.
     *
     * @param isJsonbProperty if the property has a JsonbProperty annotation
     */
    public void jsonbProperty(boolean isJsonbProperty) {
        this.isJsonbProperty = isJsonbProperty;
    }

    /**
     * Indicates if the property has a JsonbProperty annotation.
     *
     * @return true if the property has a JsonbProperty annotation
     */
    public boolean isJsonbProperty() {
        return isJsonbProperty;
    }

    @Override
    public String toString() {
        return "FieldDefinition{"
                + "name='" + name + '\''
                + ", returnType='" + returnType + '\''
                + ", isArrayReturnType=" + isArrayReturnType
                + ", isReturnTypeMandatory=" + isReturnTypeMandatory
                + ", isArrayReturnTypeMandatory=" + isArrayReturnTypeMandatory
                + ", listArguments=" + listSchemaArguments
                + ", arrayLevels=" + arrayLevels
                + ", originalType=" + originalType
                + ", defaultFormatApplied=" + defaultFormatApplied
                + ", originalArrayType=" + originalArrayType
                + ", format=" + Arrays.toString(format)
                + ", isJsonbFormat=" + isJsonbFormat
                + ", isJsonbProperty=" + isJsonbProperty
                + ", description='" + description() + '\'' + '}';
    }


    /**
     * A fluent API {@link io.helidon.common.Builder} to build instances of {@link SchemaFieldDefinition}.
     */
    public static class Builder implements io.helidon.common.Builder {
        private String name;
        private String returnType;
        private boolean isArrayReturnType;
        private int arrayLevels;
        private boolean isReturnTypeMandatory;
        private List listSchemaArguments = new ArrayList<>();
        private boolean isArrayReturnTypeMandatory;
        private DataFetcher dataFetcher;
        private Class originalType;
        private Class originalArrayType;
        private String[] format;
        private Object defaultValue;
        private boolean defaultFormatApplied;
        private boolean isJsonbFormat;
        private boolean isJsonbProperty;
        private String description;

        /**
         * Set the name of the {@link SchemaFieldDefinition}.
         *
         * @param name the name of the {@link SchemaFieldDefinition}
         * @return updated builder instance
         */
        public Builder name(String name) {
            this.name = name;
            return this;
        }

        /**
         * Set the returnType.
         *
         * @param returnType the returnType
         * @return updated builder instance
         */
        public Builder returnType(String returnType) {
            this.returnType = returnType;
            return this;
        }

        /**
         * Set if the return type is an array type such as a native array([]) or a List, Collection.
         *
         * @param isArrayReturnType true if the return type is an array type
         * @return updated builder instance
         */
        public Builder arrayReturnType(boolean isArrayReturnType) {
            this.isArrayReturnType = isArrayReturnType;
            return this;
        }

        /**
         * Set if the return type is mandatory.
         *
         * @param isReturnTypeMandatory true if the return type is mandatory.
         * @return updated builder instance
         */
        public Builder returnTypeMandatory(boolean isReturnTypeMandatory) {
            this.isReturnTypeMandatory = isReturnTypeMandatory;
            return this;
        }

        /**
         * Set the number of array levels if return type is an array.
         *
         * @param arrayLevels the number of array levels if return type is an array
         * @return updated builder instance
         */
        public Builder arrayLevels(int arrayLevels) {
            this.arrayLevels = arrayLevels;
            return this;
        }

        /**
         * Add an argument to the {@link SchemaFieldDefinition}.
         *
         * @param argument the argument to add to the {@link SchemaFieldDefinition}
         * @return updated builder instance
         */
        public Builder addArgument(SchemaArgument argument) {
            listSchemaArguments.add(argument);
            return this;
        }

        /**
         * Set if the value of the array is mandatory.
         *
         * @param isArrayReturnTypeMandatory If the return type is an array then indicates if the value in the
         *                                   array is mandatory
         * @return updated builder instance
         */
        public Builder arrayReturnTypeMandatory(boolean isArrayReturnTypeMandatory) {
            this.isArrayReturnTypeMandatory = isArrayReturnTypeMandatory;
            return this;
        }

        /**
         * Set the {@link DataFetcher} to override default behaviour of field.
         * @param dataFetcher {@link DataFetcher} to override default behaviour of field
         * @return updated builder instance
         */
        public Builder dataFetcher(DataFetcher dataFetcher) {
            this.dataFetcher = dataFetcher;
            return this;
        }

        /**
         * Set the original return type.
         * @param originalType  the original return type
         * @return updated builder instance
         */
        public Builder originalType(Class originalType) {
            this.originalType = originalType;
            return this;
        }

        /**
         * Set the original array inner type if it is array type.
         * @param originalArrayType  the  original array inner type if it is array type
         * @return updated builder instance
         */
        public Builder originalArrayType(Class originalArrayType) {
            this.originalArrayType = originalArrayType;
            return this;
        }

        /**
         * Set the format for a number or date.
         * @param format the format for a number or date
         * @return updated builder instance
         */
        public Builder format(String[] format) {
            if (format == null) {
                this.format = null;
            } else {
                this.format = new String[format.length];
                System.arraycopy(format, 0, this.format, 0, this.format.length);
            }

            return this;
        }

        /**
         * Set the default value for this field definition. Only valid for field definitions of an input type.
         * @param defaultValue the default value for this field definition
         * @return updated builder instance
         */
        public Builder defaultValue(Object defaultValue) {
            this.defaultValue = defaultValue;
            return this;
        }

        /**
         * Set if the field has a default format applied.
         * @param defaultFormatApplied true if the field has a default format applied
         * @return updated builder instance
         */
        public Builder defaultFormatApplied(boolean defaultFormatApplied) {
            this.defaultFormatApplied = defaultFormatApplied;
            return this;
        }

        /**
         * Set if the format is of type Jsonb.
         * @param isJsonbFormat if the format is of type Jsonb.
         * @return updated builder instance
         */
        public Builder jsonbFormat(boolean isJsonbFormat) {
            this.isJsonbFormat = isJsonbFormat;
            return this;
        }
        /**
         * Set if the property name is of type Jsonb.
         * @param isJsonbProperty if the property name is of type Jsonb.
         * @return updated builder instance
         */
        public Builder jsonbProperty(boolean isJsonbProperty) {
            this.isJsonbProperty = isJsonbProperty;
            return this;
        }

        /**
         * Set the description.
         * @param description the description of the {@link SchemaFieldDefinition}
         * @return updated builder instance
         */
        public Builder description(String description) {
            this.description = description;
            return this;
        }

        @Override
        public SchemaFieldDefinition build() {
            Objects.requireNonNull(name, "Name must be specified");
            return new SchemaFieldDefinition(this);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy