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

software.amazon.smithy.lsp.project.ProjectConfigLoader Maven / Gradle / Ivy

The newest version!
/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 * SPDX-License-Identifier: Apache-2.0
 */

package software.amazon.smithy.lsp.project;

import java.io.File;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.logging.Logger;
import software.amazon.smithy.build.model.SmithyBuildConfig;
import software.amazon.smithy.lsp.ServerState;
import software.amazon.smithy.lsp.document.Document;
import software.amazon.smithy.lsp.util.Result;
import software.amazon.smithy.model.node.Node;
import software.amazon.smithy.model.node.NodeMapper;
import software.amazon.smithy.model.node.ObjectNode;
import software.amazon.smithy.utils.IoUtils;

/**
 * Loads {@link ProjectConfig}s from a given root directory
 *
 * 

This aggregates configuration from multiple sources, including * {@link ProjectConfigLoader#SMITHY_BUILD}, * {@link ProjectConfigLoader#SMITHY_BUILD_EXTS}, and * {@link ProjectConfigLoader#SMITHY_PROJECT}. Each of these are looked * for in the project root directory. If none are found, an empty smithy-build * is assumed. Any exceptions that occur are aggregated and will fail the load. * *

Aggregation is done as follows: *

    *
  1. * Start with an empty {@link SmithyBuildConfig.Builder}. This will * aggregate {@link SmithyBuildConfig} and {@link SmithyBuildExtensions} *
  2. *
  3. * If a smithy-build.json exists, try to load it. If one doesn't exist, * use an empty {@link SmithyBuildConfig} (with version "1"). Merge the result * into the builder *
  4. *
  5. * If any {@link ProjectConfigLoader#SMITHY_BUILD_EXTS} exist, try to load * and merge them into a single {@link SmithyBuildExtensions.Builder} *
  6. *
  7. * If a {@link ProjectConfigLoader#SMITHY_PROJECT} exists, try to load it. * Otherwise use an empty {@link ProjectConfig.Builder}. This will be the * result of the load *
  8. *
  9. * Merge any {@link ProjectConfigLoader#SMITHY_BUILD_EXTS} into the original * {@link SmithyBuildConfig.Builder} and build it *
  10. *
  11. * Add all sources, imports, and MavenConfig from the {@link SmithyBuildConfig} * to the {@link ProjectConfig.Builder} *
  12. *
  13. * If the {@link ProjectConfig.Builder} doesn't specify an outputDirectory, * use the one in {@link SmithyBuildConfig}, if present *
  14. *
*/ public final class ProjectConfigLoader { public static final String SMITHY_BUILD = "smithy-build.json"; public static final String[] SMITHY_BUILD_EXTS = { "build" + File.separator + "smithy-dependencies.json", ".smithy.json"}; public static final String SMITHY_PROJECT = ".smithy-project.json"; public static final List PROJECT_BUILD_FILES = new ArrayList<>(2 + SMITHY_BUILD_EXTS.length); private static final Logger LOGGER = Logger.getLogger(ProjectConfigLoader.class.getName()); private static final SmithyBuildConfig DEFAULT_SMITHY_BUILD = SmithyBuildConfig.builder().version("1").build(); private static final NodeMapper NODE_MAPPER = new NodeMapper(); static { PROJECT_BUILD_FILES.add(SMITHY_BUILD); PROJECT_BUILD_FILES.add(SMITHY_PROJECT); PROJECT_BUILD_FILES.addAll(Arrays.asList(SMITHY_BUILD_EXTS)); } private ProjectConfigLoader() { } static Result> loadFromRoot(Path workspaceRoot) { return loadFromRoot(workspaceRoot, new ServerState()); } static Result> loadFromRoot(Path workspaceRoot, ServerState state) { SmithyBuildConfig.Builder builder = SmithyBuildConfig.builder(); List exceptions = new ArrayList<>(); Map buildFiles = new HashMap<>(); Path smithyBuildPath = workspaceRoot.resolve(SMITHY_BUILD); if (Files.isRegularFile(smithyBuildPath)) { LOGGER.info("Loading smithy-build.json from " + smithyBuildPath); Result result = Result.ofFallible(() -> { BuildFile buildFile = addBuildFile(buildFiles, smithyBuildPath, state); return SmithyBuildConfig.fromNode( Node.parseJsonWithComments(buildFile.document().copyText(), buildFile.path())); }); result.get().ifPresent(builder::merge); result.getErr().ifPresent(exceptions::add); } else { LOGGER.info("No smithy-build.json found at " + smithyBuildPath + ", defaulting to empty config."); builder.merge(DEFAULT_SMITHY_BUILD); } SmithyBuildExtensions.Builder extensionsBuilder = SmithyBuildExtensions.builder(); for (String ext : SMITHY_BUILD_EXTS) { Path extPath = workspaceRoot.resolve(ext); if (Files.isRegularFile(extPath)) { Result result = Result.ofFallible(() -> { BuildFile buildFile = addBuildFile(buildFiles, extPath, state); return loadSmithyBuildExtensions(buildFile); }); result.get().ifPresent(extensionsBuilder::merge); result.getErr().ifPresent(exceptions::add); } } ProjectConfig.Builder finalConfigBuilder = ProjectConfig.builder(); Path smithyProjectPath = workspaceRoot.resolve(SMITHY_PROJECT); if (Files.isRegularFile(smithyProjectPath)) { LOGGER.info("Loading .smithy-project.json from " + smithyProjectPath); Result result = Result.ofFallible(() -> { BuildFile buildFile = addBuildFile(buildFiles, smithyProjectPath, state); return ProjectConfig.Builder.load(buildFile); }); if (result.isOk()) { finalConfigBuilder = result.unwrap(); } else { exceptions.add(result.unwrapErr()); } } if (!exceptions.isEmpty()) { return Result.err(exceptions); } builder.merge(extensionsBuilder.build().asSmithyBuildConfig()); SmithyBuildConfig config = builder.build(); finalConfigBuilder.addSources(config.getSources()).addImports(config.getImports()); config.getMaven().ifPresent(finalConfigBuilder::mavenConfig); if (finalConfigBuilder.outputDirectory == null) { config.getOutputDirectory().ifPresent(finalConfigBuilder::outputDirectory); } finalConfigBuilder.buildFiles(buildFiles); return Result.ok(finalConfigBuilder.build()); } private static BuildFile addBuildFile(Map buildFiles, Path path, ServerState state) { Document managed = state.getManagedDocument(path); BuildFile buildFile; if (managed != null) { buildFile = new BuildFile(path.toString(), managed); } else { Document document = Document.of(IoUtils.readUtf8File(path)); buildFile = new BuildFile(path.toString(), document); } buildFiles.put(buildFile.path(), buildFile); return buildFile; } private static SmithyBuildExtensions loadSmithyBuildExtensions(BuildFile buildFile) { // NOTE: This is the legacy way we loaded build extensions. It used to throw a checked exception. ObjectNode node = Node.parseJsonWithComments( buildFile.document().copyText(), buildFile.path()).expectObjectNode(); SmithyBuildExtensions config = NODE_MAPPER.deserialize(node, SmithyBuildExtensions.class); config.mergeMavenFromSmithyBuildConfig(SmithyBuildConfig.fromNode(node)); return config; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy