io.helidon.microprofile.graphql.server.CustomScalars Maven / Gradle / Ivy
/*
* Copyright (c) 2020, 2022 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.math.BigDecimal;
import java.math.BigInteger;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.OffsetDateTime;
import java.time.OffsetTime;
import java.time.ZonedDateTime;
import java.util.Arrays;
import java.util.Date;
import graphql.language.StringValue;
import graphql.scalars.ExtendedScalars;
import graphql.schema.Coercing;
import graphql.schema.CoercingParseLiteralException;
import graphql.schema.CoercingParseValueException;
import graphql.schema.CoercingSerializeException;
import graphql.schema.GraphQLScalarType;
import static graphql.Scalars.GraphQLFloat;
import static graphql.Scalars.GraphQLInt;
import static graphql.scalars.ExtendedScalars.GraphQLBigDecimal;
import static graphql.scalars.ExtendedScalars.GraphQLBigInteger;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.DATETIME_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.DATE_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.FORMATTED_DATETIME_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.FORMATTED_DATE_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.FORMATTED_OFFSET_DATETIME_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.FORMATTED_TIME_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.FORMATTED_ZONED_DATETIME_SCALAR;
import static io.helidon.microprofile.graphql.server.SchemaGeneratorHelper.TIME_SCALAR;
/**
* Custom scalars.
*/
class CustomScalars {
/**
* Private no-args constructor.
*/
private CustomScalars() {
}
/**
* An instance of a custome BigDecimal Scalar.
*/
static final GraphQLScalarType CUSTOM_BIGDECIMAL_SCALAR = newCustomBigDecimalScalar();
/**
* An instance of a custom Int scalar.
*/
static final GraphQLScalarType CUSTOM_INT_SCALAR = newCustomGraphQLInt();
/**
* An instance of a custom Float scalar.
*/
static final GraphQLScalarType CUSTOM_FLOAT_SCALAR = newCustomGraphQLFloat();
/**
* An instance of a custom BigInteger scalar.
*/
static final GraphQLScalarType CUSTOM_BIGINTEGER_SCALAR = newCustomGraphQLBigInteger();
/**
* An instance of a custom formatted date/time scalar.
*/
static final GraphQLScalarType FORMATTED_CUSTOM_DATE_TIME_SCALAR = newDateTimeScalar(FORMATTED_DATETIME_SCALAR);
/**
* An instance of a custom formatted time scalar.
*/
static final GraphQLScalarType FORMATTED_CUSTOM_TIME_SCALAR = newTimeScalar(FORMATTED_TIME_SCALAR);
/**
* An instance of a custom formatted date scalar.
*/
static final GraphQLScalarType FORMATTED_CUSTOM_DATE_SCALAR = newDateScalar(FORMATTED_DATE_SCALAR);
/**
* An instance of a custom date/time scalar (with default formatting).
*/
static final GraphQLScalarType CUSTOM_DATE_TIME_SCALAR = newDateTimeScalar(DATETIME_SCALAR);
/**
* An instance of a custom offset date/time scalar (with default formatting).
*/
static final GraphQLScalarType CUSTOM_OFFSET_DATE_TIME_SCALAR =
newOffsetDateTimeScalar(FORMATTED_OFFSET_DATETIME_SCALAR);
/**
* An instance of a custom offset date/time scalar (with default formatting).
*/
static final GraphQLScalarType CUSTOM_ZONED_DATE_TIME_SCALAR =
newZonedDateTimeScalar(FORMATTED_ZONED_DATETIME_SCALAR);
/**
* An instance of a custom time scalar (with default formatting).
*/
static final GraphQLScalarType CUSTOM_TIME_SCALAR = newTimeScalar(TIME_SCALAR);
/**
* An instance of a custom date scalar (with default formatting).
*/
static final GraphQLScalarType CUSTOM_DATE_SCALAR = newDateScalar(DATE_SCALAR);
/**
* Return a new custom date/time scalar.
*
* @param name the name of the scalar
* @return a new custom date/time scalar
*/
static GraphQLScalarType newDateTimeScalar(String name) {
GraphQLScalarType originalScalar = ExtendedScalars.DateTime;
return GraphQLScalarType.newScalar()
.coercing(new DateTimeCoercing())
.name(name)
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom offset date/time scalar.
*
* @param name the name of the scalar
* @return a new custom date/time scalar
*/
@SuppressWarnings("unchecked")
static GraphQLScalarType newOffsetDateTimeScalar(String name) {
GraphQLScalarType originalScalar = ExtendedScalars.DateTime;
return GraphQLScalarType.newScalar()
.coercing(new DateTimeCoercing())
.name(name)
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom zoned date/time scalar.
*
* @param name the name of the scalar
* @return a new custom date/time scalar
*/
@SuppressWarnings("unchecked")
static GraphQLScalarType newZonedDateTimeScalar(String name) {
GraphQLScalarType originalScalar = ExtendedScalars.DateTime;
return GraphQLScalarType.newScalar()
.coercing(new DateTimeCoercing())
.name(name)
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom time scalar.
*
* @param name the name of the scalar
* @return a new custom time scalar
*/
static GraphQLScalarType newTimeScalar(String name) {
GraphQLScalarType originalScalar = ExtendedScalars.Time;
return GraphQLScalarType.newScalar()
.coercing(new TimeCoercing())
.name(name)
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom date scalar.
*
* @param name the name of the scalar
* @return a new custom date scalar
*/
static GraphQLScalarType newDateScalar(String name) {
GraphQLScalarType originalScalar = ExtendedScalars.Date;
return GraphQLScalarType.newScalar()
.coercing(new DateCoercing())
.name(name)
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom BigDecimal scalar.
*
* @return a new custom BigDecimal scalar
*/
private static GraphQLScalarType newCustomBigDecimalScalar() {
GraphQLScalarType originalScalar = GraphQLBigDecimal;
return GraphQLScalarType.newScalar()
.coercing(new NumberCoercing(originalScalar.getCoercing()))
.name(originalScalar.getName())
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom Int scalar.
*
* @return a new custom Int scalar
*/
private static GraphQLScalarType newCustomGraphQLInt() {
GraphQLScalarType originalScalar = GraphQLInt;
return GraphQLScalarType.newScalar()
.coercing(new NumberCoercing(originalScalar.getCoercing()))
.name(originalScalar.getName())
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom Float scalar.
*
* @return a new custom Float scalar
*/
private static GraphQLScalarType newCustomGraphQLFloat() {
GraphQLScalarType originalScalar = GraphQLFloat;
return GraphQLScalarType.newScalar()
.coercing(new NumberCoercing(originalScalar.getCoercing()))
.name(originalScalar.getName())
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Return a new custom BigInteger scalar.
*
* @return a new custom BigInteger scalar
*/
private static GraphQLScalarType newCustomGraphQLBigInteger() {
GraphQLScalarType originalScalar = GraphQLBigInteger;
return GraphQLScalarType.newScalar()
.coercing(new NumberCoercing(originalScalar.getCoercing()))
.name(originalScalar.getName())
.description("Custom: " + originalScalar.getDescription())
.build();
}
/**
* Abstract implementation of {@link Coercing} interface for given classes.
*/
abstract static class AbstractDateTimeCoercing implements Coercing {
/**
* {@link Class}es that can be coerced.
*/
private final Class>[] clazzes;
/**
* Construct a {@link AbstractDateTimeCoercing}.
*
* @param clazzes {@link Class}es to coerce
*/
AbstractDateTimeCoercing(Class>... clazzes) {
this.clazzes = clazzes;
}
@Override
public Object serialize(Object dataFetcherResult) throws CoercingSerializeException {
return convert(dataFetcherResult);
}
@Override
public Object parseValue(Object input) throws CoercingParseValueException {
return convert(input);
}
@Override
public Object parseLiteral(Object input) throws CoercingParseLiteralException {
return parseStringLiteral(input);
}
/**
* Convert the given input to the type of if a String then leave it be.
*
* @param input input to coerce
* @return the coerced value
* @throws CoercingParseLiteralException if any exceptions converting
*/
private Object convert(Object input) throws CoercingParseLiteralException {
if (input instanceof String) {
return input;
}
for (Class> clazz : clazzes) {
if (clazz.isInstance(input)) {
return input;
}
}
throw new CoercingParseLiteralException("Unable to convert type of " + input.getClass().toString()
+ " with classes " + Arrays.toString(clazzes));
}
/**
* Parse a String literal and return instance of {@link StringValue} or throw an exception.
*
* @param input input to parse
* @throws CoercingParseLiteralException if it is not a {@link StringValue}
*/
private String parseStringLiteral(Object input) throws CoercingParseLiteralException {
if (!(input instanceof StringValue)) {
throw new CoercingParseLiteralException("Expected AST type 'StringValue' but was '"
+ (
input == null
? "null"
: input.getClass().getSimpleName()) + "'.");
}
return ((StringValue) input).getValue();
}
}
/**
* Coercing Implementation for Date/Time.
*/
static class DateTimeCoercing extends AbstractDateTimeCoercing {
/**
* Construct a {@link DateTimeCoercing}.
*/
DateTimeCoercing() {
super(LocalDateTime.class, OffsetDateTime.class, ZonedDateTime.class);
}
}
/**
* Coercing implementation for Time.
*/
static class TimeCoercing extends AbstractDateTimeCoercing {
/**
* Construct a {@link TimeCoercing}.
*/
TimeCoercing() {
super(LocalTime.class, OffsetTime.class);
}
}
/**
* Coercing implementation for Date.
*/
static class DateCoercing extends AbstractDateTimeCoercing {
/**
* Construct a {@link DateCoercing}.
*/
DateCoercing() {
super(LocalDate.class, Date.class);
}
}
/**
* Coercing implementation for BigDecimal.
*/
static class BigDecimalCoercing extends AbstractDateTimeCoercing {
/**
* Construct a {@link BigDecimalCoercing}.
*/
BigDecimalCoercing() {
super(BigDecimal.class);
}
}
/**
* Number implementation of {@link Coercing} interface for given classes.
* @param defines input type
*/
@SuppressWarnings("unchecked")
static class NumberCoercing implements Coercing {
/**
* Original {@link Coercing} to fall back on if neeed.
*/
private final Coercing originalCoercing;
/**
* Construct a {@link NumberCoercing} from an original {@link Coercing}.
*
* @param originalCoercing original {@link Coercing}
*/
NumberCoercing(Coercing originalCoercing) {
this.originalCoercing = originalCoercing;
}
@Override
public Object serialize(Object dataFetcherResult) throws CoercingSerializeException {
return dataFetcherResult instanceof String
? (String) dataFetcherResult
: originalCoercing.serialize(dataFetcherResult);
}
@Override
public I parseValue(Object input) throws CoercingParseValueException {
return (I) originalCoercing.parseValue(input);
}
@Override
public I parseLiteral(Object input) throws CoercingParseLiteralException {
return (I) originalCoercing.parseLiteral(input);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy