GraphQLCodegen.java 21.1 KB
Newer Older
1 2
package com.kobylynskyi.graphql.codegen;

3 4 5 6 7 8 9 10
import com.kobylynskyi.graphql.codegen.mapper.EnumDefinitionToDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.FieldDefinitionToParameterMapper;
import com.kobylynskyi.graphql.codegen.mapper.FieldDefinitionsToResolverDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.InputDefinitionToDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.InterfaceDefinitionToDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.RequestResponseDefinitionToDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.TypeDefinitionToDataModelMapper;
import com.kobylynskyi.graphql.codegen.mapper.UnionDefinitionToDataModelMapper;
11 12
import com.kobylynskyi.graphql.codegen.model.ApiNamePrefixStrategy;
import com.kobylynskyi.graphql.codegen.model.ApiRootInterfaceStrategy;
13
import com.kobylynskyi.graphql.codegen.model.GeneratedInformation;
14
import com.kobylynskyi.graphql.codegen.model.MappingConfig;
15
import com.kobylynskyi.graphql.codegen.model.MappingConfigConstants;
16
import com.kobylynskyi.graphql.codegen.model.MappingContext;
17 18 19 20 21 22 23 24
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedDocument;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedEnumTypeDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedFieldDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedInputObjectTypeDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedInterfaceTypeDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedObjectTypeDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedScalarTypeDefinition;
import com.kobylynskyi.graphql.codegen.model.definitions.ExtendedUnionTypeDefinition;
A
Alberto Valiña 已提交
25
import com.kobylynskyi.graphql.codegen.supplier.MappingConfigSupplier;
26
import com.kobylynskyi.graphql.codegen.utils.Utils;
27 28
import graphql.language.FieldDefinition;
import graphql.language.ScalarTypeExtensionDefinition;
29 30 31 32
import lombok.Getter;
import lombok.Setter;

import java.io.File;
B
Bogdan Kobylynskyi 已提交
33
import java.io.IOException;
34 35 36 37 38 39
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Optional;
40

41 42
import static java.util.stream.Collectors.toList;

43 44
/**
 * Generator of:
45 46 47
 * - Interface for each GraphQL query, mutation, subscription, union and field resolvers
 * - POJO Class for each GraphQL type and input
 * - Enum Class for each GraphQL enum
48 49
 *
 * @author kobylynskyi
A
Alberto Valiña 已提交
50
 * @author valinhadev
51 52 53
 */
@Getter
@Setter
54
public class GraphQLCodegen {
55 56 57 58

    private List<String> schemas;
    private File outputDir;
    private MappingConfig mappingConfig;
59
    private GeneratedInformation generatedInformation;
60

61 62 63 64 65
    public GraphQLCodegen(List<String> schemas,
                          File outputDir,
                          MappingConfig mappingConfig,
                          GeneratedInformation generatedInformation) {
        this(schemas, outputDir, mappingConfig, null, generatedInformation);
A
Alberto Valiña 已提交
66 67
    }

68 69 70 71 72 73 74 75 76 77 78 79
    public GraphQLCodegen(List<String> schemas,
                          File outputDir,
                          MappingConfig mappingConfig,
                          MappingConfigSupplier externalMappingConfigSupplier) {
        this(schemas, outputDir, mappingConfig, externalMappingConfigSupplier, new GeneratedInformation());
    }

    public GraphQLCodegen(List<String> schemas,
                          File outputDir,
                          MappingConfig mappingConfig,
                          MappingConfigSupplier externalMappingConfigSupplier,
                          GeneratedInformation generatedInformation) {
80 81 82
        this.schemas = schemas;
        this.outputDir = outputDir;
        this.mappingConfig = mappingConfig;
A
Alberto Valiña 已提交
83 84
        this.mappingConfig.combine(externalMappingConfigSupplier != null ? externalMappingConfigSupplier.get() : null);
        initDefaultValues(mappingConfig);
85
        validateConfigs(mappingConfig);
86
        this.generatedInformation = generatedInformation;
A
Alberto Valiña 已提交
87 88
    }

89
    private static void initDefaultValues(MappingConfig mappingConfig) {
A
Alberto Valiña 已提交
90
        if (mappingConfig.getModelValidationAnnotation() == null) {
91
            mappingConfig.setModelValidationAnnotation(MappingConfigConstants.DEFAULT_VALIDATION_ANNOTATION);
A
Alberto Valiña 已提交
92
        }
93
        if (mappingConfig.getGenerateBuilder() == null) {
94
            mappingConfig.setGenerateBuilder(MappingConfigConstants.DEFAULT_BUILDER);
95
        }
A
Alberto Valiña 已提交
96
        if (mappingConfig.getGenerateEqualsAndHashCode() == null) {
97
            mappingConfig.setGenerateEqualsAndHashCode(MappingConfigConstants.DEFAULT_EQUALS_AND_HASHCODE);
A
Alberto Valiña 已提交
98
        }
99 100
        if (mappingConfig.getGenerateClient() == null) {
            mappingConfig.setGenerateClient(MappingConfigConstants.DEFAULT_GENERATE_CLIENT);
101 102
        }
        if (mappingConfig.getRequestSuffix() == null) {
103
            mappingConfig.setRequestSuffix(MappingConfigConstants.DEFAULT_REQUEST_SUFFIX);
104
        }
105 106 107
        if (mappingConfig.getResponseSuffix() == null) {
            mappingConfig.setResponseSuffix(MappingConfigConstants.DEFAULT_RESPONSE_SUFFIX);
        }
108
        if (mappingConfig.getResponseProjectionSuffix() == null) {
109
            mappingConfig.setResponseProjectionSuffix(MappingConfigConstants.DEFAULT_RESPONSE_PROJECTION_SUFFIX);
110
        }
111
        if (mappingConfig.getParametrizedInputSuffix() == null) {
112
            mappingConfig.setParametrizedInputSuffix(MappingConfigConstants.DEFAULT_PARAMETRIZED_INPUT_SUFFIX);
113
        }
114 115 116
        if (mappingConfig.getGenerateImmutableModels() == null) {
            mappingConfig.setGenerateImmutableModels(MappingConfigConstants.DEFAULT_GENERATE_IMMUTABLE_MODELS);
        }
A
Alberto Valiña 已提交
117
        if (mappingConfig.getGenerateToString() == null) {
118
            mappingConfig.setGenerateToString(MappingConfigConstants.DEFAULT_TO_STRING);
A
Alberto Valiña 已提交
119 120
        }
        if (mappingConfig.getGenerateApis() == null) {
121
            mappingConfig.setGenerateApis(MappingConfigConstants.DEFAULT_GENERATE_APIS);
A
Alberto Valiña 已提交
122
        }
123
        if (mappingConfig.getApiNameSuffix() == null) {
124 125 126 127
            mappingConfig.setApiNameSuffix(MappingConfigConstants.DEFAULT_RESOLVER_SUFFIX);
        }
        if (mappingConfig.getTypeResolverSuffix() == null) {
            mappingConfig.setTypeResolverSuffix(MappingConfigConstants.DEFAULT_RESOLVER_SUFFIX);
128
        }
129
        if (mappingConfig.getGenerateAsyncApi() == null) {
130
            mappingConfig.setGenerateAsyncApi(MappingConfigConstants.DEFAULT_GENERATE_ASYNC_APIS);
131
        }
132
        if (mappingConfig.getGenerateParameterizedFieldsResolvers() == null) {
133
            mappingConfig.setGenerateParameterizedFieldsResolvers(MappingConfigConstants.DEFAULT_GENERATE_PARAMETERIZED_FIELDS_RESOLVERS);
134
        }
135
        if (mappingConfig.getGenerateExtensionFieldsResolvers() == null) {
136
            mappingConfig.setGenerateExtensionFieldsResolvers(MappingConfigConstants.DEFAULT_GENERATE_EXTENSION_FIELDS_RESOLVERS);
137
        }
138
        if (mappingConfig.getGenerateDataFetchingEnvironmentArgumentInApis() == null) {
139
            mappingConfig.setGenerateDataFetchingEnvironmentArgumentInApis(MappingConfigConstants.DEFAULT_GENERATE_DATA_FETCHING_ENV);
140
        }
141 142 143
        if (mappingConfig.getGenerateModelsForRootTypes() == null) {
            mappingConfig.setGenerateModelsForRootTypes(MappingConfigConstants.DEFAULT_GENERATE_MODELS_FOR_ROOT_TYPES);
        }
144 145 146 147 148 149
        if (mappingConfig.getApiNamePrefixStrategy() == null) {
            mappingConfig.setApiNamePrefixStrategy(MappingConfigConstants.DEFAULT_API_NAME_PREFIX_STRATEGY);
        }
        if (mappingConfig.getApiRootInterfaceStrategy() == null) {
            mappingConfig.setApiRootInterfaceStrategy(MappingConfigConstants.DEFAULT_API_ROOT_INTERFACE_STRATEGY);
        }
B
Bogdan Kobylynskyi 已提交
150
        if (Boolean.TRUE.equals(mappingConfig.getGenerateClient())) {
151 152 153
            // required for request serialization
            mappingConfig.setGenerateToString(true);
        }
154 155 156
    }

    private static void validateConfigs(MappingConfig mappingConfig) {
157 158
        if (mappingConfig.getApiRootInterfaceStrategy() == ApiRootInterfaceStrategy.INTERFACE_PER_SCHEMA &&
                mappingConfig.getApiNamePrefixStrategy() == ApiNamePrefixStrategy.CONSTANT) {
159
            // we will have a conflict in case there is "type Query" in multiple graphql schema files
160 161
            throw new IllegalArgumentException("API prefix should not be CONSTANT for INTERFACE_PER_SCHEMA option");
        }
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
        if (mappingConfig.getGenerateApis() &&
                mappingConfig.getGenerateModelsForRootTypes() &&
                mappingConfig.getApiNamePrefixStrategy() == ApiNamePrefixStrategy.CONSTANT) {
            // checking for conflict between root type model classes and api interfaces
            if (Utils.stringsEqualIgnoreSpaces(mappingConfig.getApiNamePrefix(), mappingConfig.getModelNamePrefix()) &&
                    Utils.stringsEqualIgnoreSpaces(mappingConfig.getApiNameSuffix(), mappingConfig.getModelNameSuffix())) {
                // we will have a conflict between model pojo (Query.java) and api interface (Query.java)
                throw new IllegalArgumentException("Either disable APIs generation or set different Prefix/Suffix for API classes and model classes");
            }
            // checking for conflict between root type model resolver classes and api interfaces
            if (Utils.stringsEqualIgnoreSpaces(mappingConfig.getApiNamePrefix(), mappingConfig.getTypeResolverPrefix()) &&
                    Utils.stringsEqualIgnoreSpaces(mappingConfig.getApiNameSuffix(), mappingConfig.getTypeResolverSuffix())) {
                // we will have a conflict between model resolver interface (QueryResolver.java) and api interface resolver (QueryResolver.java)
                throw new IllegalArgumentException("Either disable APIs generation or set different Prefix/Suffix for API classes and type resolver classes");
            }
        }
178 179
    }

B
Bogdan Kobylynskyi 已提交
180
    public List<File> generate() throws IOException {
181
        GraphQLCodegenFileCreator.prepareOutputDir(outputDir);
182
        long startTime = System.currentTimeMillis();
183
        List<File> generatedFiles = Collections.emptyList();
184
        if (!schemas.isEmpty()) {
185
            ExtendedDocument document = GraphQLDocumentParser.getDocument(mappingConfig, schemas);
186
            initCustomTypeMappings(document.getScalarDefinitions());
187
            generatedFiles = processDefinitions(document);
188
        }
189
        long elapsed = System.currentTimeMillis() - startTime;
190
        System.out.println(String.format("Finished processing %d schema(s) in %d ms", schemas.size(), elapsed));
191
        return generatedFiles;
192 193
    }

194
    private List<File> processDefinitions(ExtendedDocument document) {
195
        MappingContext context = new MappingContext(mappingConfig, document, generatedInformation);
196

197 198
        List<File> generatedFiles = new ArrayList<>();
        for (ExtendedObjectTypeDefinition extendedObjectTypeDefinition : document.getTypeDefinitions()) {
199
            generatedFiles.addAll(generateType(context, extendedObjectTypeDefinition));
200 201
        }
        for (ExtendedObjectTypeDefinition extendedObjectTypeDefinition : document.getTypeDefinitions()) {
202
            generateFieldResolver(context, extendedObjectTypeDefinition.getFieldDefinitions(), extendedObjectTypeDefinition.getName())
203 204 205
                    .ifPresent(generatedFiles::add);
        }
        for (ExtendedObjectTypeDefinition extendedObjectTypeDefinition : document.getOperationDefinitions()) {
B
Bogdan Kobylynskyi 已提交
206
            if (Boolean.TRUE.equals(mappingConfig.getGenerateApis())) {
207 208
                generatedFiles.addAll(generateServerOperations(context, extendedObjectTypeDefinition));
            }
B
Bogdan Kobylynskyi 已提交
209
            if (Boolean.TRUE.equals(mappingConfig.getGenerateClient())) {
210 211
                generatedFiles.addAll(generateClient(context, extendedObjectTypeDefinition));
            }
212 213
        }
        for (ExtendedInputObjectTypeDefinition extendedInputObjectTypeDefinition : document.getInputDefinitions()) {
214
            generatedFiles.add(generateInput(context, extendedInputObjectTypeDefinition));
215 216
        }
        for (ExtendedEnumTypeDefinition extendedEnumTypeDefinition : document.getEnumDefinitions()) {
217
            generatedFiles.add(generateEnum(context, extendedEnumTypeDefinition));
218 219
        }
        for (ExtendedUnionTypeDefinition extendedUnionTypeDefinition : document.getUnionDefinitions()) {
220
            generatedFiles.add(generateUnion(context, extendedUnionTypeDefinition));
221 222
        }
        for (ExtendedInterfaceTypeDefinition extendedInterfaceTypeDefinition : document.getInterfaceDefinitions()) {
223
            generatedFiles.add(generateInterface(context, extendedInterfaceTypeDefinition));
224 225
        }
        for (ExtendedInterfaceTypeDefinition definition : document.getInterfaceDefinitions()) {
226
            generateFieldResolver(context, definition.getFieldDefinitions(), definition.getName())
227 228 229 230 231
                    .ifPresent(generatedFiles::add);
        }
        System.out.println(String.format("Generated %d definition classes in folder %s",
                generatedFiles.size(), outputDir.getAbsolutePath()));
        return generatedFiles;
232 233
    }

234 235
    private File generateUnion(MappingContext mappingContext, ExtendedUnionTypeDefinition definition) {
        Map<String, Object> dataModel = UnionDefinitionToDataModelMapper.map(mappingContext, definition);
236
        return GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.unionTemplate, dataModel, outputDir);
237 238
    }

239 240
    private File generateInterface(MappingContext mappingContext, ExtendedInterfaceTypeDefinition definition) {
        Map<String, Object> dataModel = InterfaceDefinitionToDataModelMapper.map(mappingContext, definition);
241
        return GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.interfaceTemplate, dataModel, outputDir);
242 243
    }

244
    private List<File> generateServerOperations(MappingContext mappingContext, ExtendedObjectTypeDefinition definition) {
245
        List<File> generatedFiles = new ArrayList<>();
246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
        // Generate a root interface with all operations inside
        // Relates to https://github.com/facebook/relay/issues/112
        switch (mappingContext.getApiRootInterfaceStrategy()) {
            case INTERFACE_PER_SCHEMA:
                for (ExtendedObjectTypeDefinition defInFile : definition.groupBySourceLocationFile().values()) {
                    generatedFiles.add(generateRootApi(mappingContext, defInFile));
                }
                break;
            case SINGLE_INTERFACE:
            default:
                generatedFiles.add(generateRootApi(mappingContext, definition));
                break;
        }

        // Generate separate interfaces for all queries, mutations and subscriptions
261
        List<String> fieldNames = definition.getFieldDefinitions().stream().map(FieldDefinition::getName).collect(toList());
262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
        switch (mappingContext.getApiNamePrefixStrategy()) {
            case FOLDER_NAME_AS_PREFIX:
                for (ExtendedObjectTypeDefinition fileDef : definition.groupBySourceLocationFolder().values()) {
                    generatedFiles.addAll(generateApis(mappingContext, fileDef, fieldNames));
                }
                break;
            case FILE_NAME_AS_PREFIX:
                for (ExtendedObjectTypeDefinition fileDef : definition.groupBySourceLocationFile().values()) {
                    generatedFiles.addAll(generateApis(mappingContext, fileDef, fieldNames));
                }
                break;
            case CONSTANT:
            default:
                generatedFiles.addAll(generateApis(mappingContext, definition, fieldNames));
                break;
277
        }
278 279
        return generatedFiles;
    }
280

281 282 283 284 285 286
    private List<File> generateClient(MappingContext mappingContext, ExtendedObjectTypeDefinition definition) {
        List<File> generatedFiles = new ArrayList<>();
        List<String> fieldNames = definition.getFieldDefinitions().stream().map(FieldDefinition::getName).collect(toList());
        for (ExtendedFieldDefinition operationDef : definition.getFieldDefinitions()) {
            Map<String, Object> requestDataModel = RequestResponseDefinitionToDataModelMapper.mapRequest(mappingContext, operationDef, definition.getName(), fieldNames);
            generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.requestTemplate, requestDataModel, outputDir));
287

288 289
            Map<String, Object> responseDataModel = RequestResponseDefinitionToDataModelMapper.mapResponse(mappingContext, operationDef, definition.getName(), fieldNames);
            generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.responseTemplate, responseDataModel, outputDir));
290
        }
291
        return generatedFiles;
292 293
    }

294 295 296 297 298 299 300 301 302 303 304 305 306 307
    private List<File> generateApis(MappingContext mappingContext, ExtendedObjectTypeDefinition definition, List<String> fieldNames) {
        List<File> generatedFiles = new ArrayList<>();
        for (ExtendedFieldDefinition operationDef : definition.getFieldDefinitions()) {
            Map<String, Object> dataModel = FieldDefinitionsToResolverDataModelMapper.mapRootTypeField(mappingContext, operationDef, definition.getName(), fieldNames);
            generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.operationsTemplate, dataModel, outputDir));
        }
        return generatedFiles;
    }

    private File generateRootApi(MappingContext mappingContext, ExtendedObjectTypeDefinition definition) {
        Map<String, Object> dataModel = FieldDefinitionsToResolverDataModelMapper.mapRootTypeFields(mappingContext, definition);
        return GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.operationsTemplate, dataModel, outputDir);
    }

308
    private List<File> generateType(MappingContext mappingContext, ExtendedObjectTypeDefinition definition) {
309
        List<File> generatedFiles = new ArrayList<>();
310
        Map<String, Object> dataModel = TypeDefinitionToDataModelMapper.map(mappingContext, definition);
311
        generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.typeTemplate, dataModel, outputDir));
312

B
Bogdan Kobylynskyi 已提交
313
        if (Boolean.TRUE.equals(mappingConfig.getGenerateClient())) {
314
            Map<String, Object> responseProjDataModel = RequestResponseDefinitionToDataModelMapper.mapResponseProjection(mappingContext, definition);
315
            generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.responseProjectionTemplate, responseProjDataModel, outputDir));
316 317 318

            for (ExtendedFieldDefinition fieldDefinition : definition.getFieldDefinitions()) {
                if (!Utils.isEmpty(fieldDefinition.getInputValueDefinitions())) {
319
                    Map<String, Object> fieldProjDataModel = RequestResponseDefinitionToDataModelMapper.mapParametrizedInput(mappingContext, fieldDefinition, definition);
320 321 322
                    generatedFiles.add(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.parametrizedInputTemplate, fieldProjDataModel, outputDir));
                }
            }
323
        }
324
        return generatedFiles;
325 326
    }

327
    private Optional<File> generateFieldResolver(MappingContext mappingContext, List<ExtendedFieldDefinition> fieldDefinitions, String definitionName) {
328
        List<ExtendedFieldDefinition> fieldDefsWithResolvers = fieldDefinitions.stream()
329
                .filter(fieldDef -> FieldDefinitionToParameterMapper.generateResolversForField(mappingContext, fieldDef, definitionName))
330 331
                .collect(toList());
        if (!fieldDefsWithResolvers.isEmpty()) {
332
            Map<String, Object> dataModel = FieldDefinitionsToResolverDataModelMapper.mapToTypeResolver(mappingContext, fieldDefsWithResolvers, definitionName);
333
            return Optional.of(GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.operationsTemplate, dataModel, outputDir));
334
        }
335
        return Optional.empty();
336 337
    }

338 339
    private File generateInput(MappingContext mappingContext, ExtendedInputObjectTypeDefinition definition) {
        Map<String, Object> dataModel = InputDefinitionToDataModelMapper.map(mappingContext, definition);
340
        return GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.typeTemplate, dataModel, outputDir);
341 342
    }

343 344
    private File generateEnum(MappingContext mappingContext, ExtendedEnumTypeDefinition definition) {
        Map<String, Object> dataModel = EnumDefinitionToDataModelMapper.map(mappingContext, definition);
345
        return GraphQLCodegenFileCreator.generateFile(FreeMarkerTemplatesRegistry.enumTemplate, dataModel, outputDir);
346 347
    }

348 349
    private void initCustomTypeMappings(Collection<ExtendedScalarTypeDefinition> scalarTypeDefinitions) {
        for (ExtendedScalarTypeDefinition definition : scalarTypeDefinitions) {
350 351 352
            if (definition.getDefinition() != null) {
                mappingConfig.putCustomTypeMappingIfAbsent(definition.getDefinition().getName(), "String");
            }
353 354
            for (ScalarTypeExtensionDefinition extension : definition.getExtensions()) {
                mappingConfig.putCustomTypeMappingIfAbsent(extension.getName(), "String");
355 356
            }
        }
357 358 359 360 361
        mappingConfig.putCustomTypeMappingIfAbsent("ID", "String");
        mappingConfig.putCustomTypeMappingIfAbsent("String", "String");
        mappingConfig.putCustomTypeMappingIfAbsent("Int", "Integer");
        mappingConfig.putCustomTypeMappingIfAbsent("Float", "Double");
        mappingConfig.putCustomTypeMappingIfAbsent("Boolean", "Boolean");
362
    }
363

364
}