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

tech.jhipster.lite.module.infrastructure.secondary.YamlFileSpringPropertiesHandler Maven / Gradle / Ivy

There is a newer version: 1.23.0
Show newest version
package tech.jhipster.lite.module.infrastructure.secondary;

import static org.yaml.snakeyaml.comments.CommentType.BLOCK;

import java.io.*;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.function.Function;
import java.util.function.Predicate;
import org.yaml.snakeyaml.*;
import org.yaml.snakeyaml.DumperOptions.FlowStyle;
import org.yaml.snakeyaml.comments.CommentLine;
import org.yaml.snakeyaml.constructor.Constructor;
import org.yaml.snakeyaml.nodes.*;
import org.yaml.snakeyaml.representer.Representer;
import tech.jhipster.lite.module.domain.Indentation;
import tech.jhipster.lite.module.domain.javaproperties.*;
import tech.jhipster.lite.shared.error.domain.Assert;
import tech.jhipster.lite.shared.error.domain.GeneratorException;
import tech.jhipster.lite.shared.generation.domain.ExcludeFromGeneratedCodeCoverage;

class YamlFileSpringPropertiesHandler {

  private static final Map, Tag> TAG_BY_JAVA_TYPE = buildTagByJavaType();

  private final Path file;
  private final Indentation indentation;
  private final Yaml yaml;

  public YamlFileSpringPropertiesHandler(Path file, Indentation indentation) {
    this.indentation = indentation;
    Assert.notNull("file", file);

    this.file = file;
    this.yaml = createYaml();
  }

  private static Map, Tag> buildTagByJavaType() {
    return Map.of(Integer.class, Tag.INT, Long.class, Tag.INT, Double.class, Tag.FLOAT, Float.class, Tag.FLOAT, Boolean.class, Tag.BOOL);
  }

  @ExcludeFromGeneratedCodeCoverage(reason = "Hard to cover IOException")
  public void setValue(PropertyKey key, PropertyValue value) {
    Assert.notNull("key", key);
    Assert.notNull("value", value);

    try {
      MappingNode configuration = loadConfiguration(file.toFile());
      appendPropertyToConfiguration(key, value, configuration);
      saveConfiguration(configuration);
    } catch (IOException exception) {
      throw GeneratorException.technicalError("Error updating Yaml properties: " + exception.getMessage(), exception);
    }
  }

  @ExcludeFromGeneratedCodeCoverage(reason = "Hard to cover IOException")
  public void setComment(PropertyKey key, Comment comment) {
    Assert.notNull("key", key);
    Assert.notNull("comment", comment);

    try {
      MappingNode configuration = loadConfiguration(file.toFile());
      addCommentToConfiguration(key, comment, configuration);
      saveConfiguration(configuration);
    } catch (IOException exception) {
      throw GeneratorException.technicalError("Error updating Yaml properties: " + exception.getMessage(), exception);
    }
  }

  private void appendPropertyToConfiguration(PropertyKey key, PropertyValue value, MappingNode configuration) {
    String localKey = extractKeysParts(key).getLast();

    List parentValue = parentPropertyNode(key, configuration).getValue();

    NodeTuple nodeProperty = new NodeTuple(buildScalarNode(localKey), buildValueNode(value));
    indexInCollection(parentValue, localKey).ifPresentOrElse(
      index -> parentValue.set(index, nodeProperty),
      () -> parentValue.add(nodeProperty)
    );
  }

  private Optional indexInCollection(List nodesTuples, String key) {
    Predicate matchesKey = nodeTupleKeyEquals(key);
    for (int i = 0; i < nodesTuples.size(); i++) {
      if (matchesKey.test(nodesTuples.get(i))) {
        return Optional.of(i);
      }
    }
    return Optional.empty();
  }

  private Node buildValueNode(PropertyValue value) {
    if (value.values().size() > 1) {
      List nodes = value.values().stream().map(this::buildScalarNode).toList();

      return new SequenceNode(Tag.SEQ, nodes, FlowStyle.AUTO);
    }

    return buildScalarNode(value.values().iterator().next());
  }

  private void addCommentToConfiguration(PropertyKey key, Comment comment, MappingNode configuration) {
    String localKey = extractKeysParts(key).getLast();

    parentPropertyNode(key, configuration)
      .getValue()
      .stream()
      .filter(nodeTupleKeyEquals(localKey))
      .map(NodeTuple::getKeyNode)
      .findFirst()
      .ifPresent(keyNode -> keyNode.setBlockComments(commentLines(comment)));
  }

  private List commentLines(Comment comment) {
    return splitLines(comment).stream().map(commentLine -> new CommentLine(null, null, " " + commentLine, BLOCK)).toList();
  }

  private static Collection splitLines(Comment comment) {
    return List.of(comment.get().split("\\r?\\n"));
  }

  private MappingNode parentPropertyNode(PropertyKey key, MappingNode configuration) {
    List allKeys = extractKeysParts(key);
    List parentKeys = allKeys.subList(0, allKeys.size() - 1);

    MappingNode parentMappingNode = configuration;

    for (String partialKey : parentKeys) {
      parentMappingNode = findPartialKeyMappingNode(parentMappingNode, partialKey);
    }

    return parentMappingNode;
  }

  private MappingNode findPartialKeyMappingNode(MappingNode parentMappingNode, String partialKey) {
    return parentMappingNode
      .getValue()
      .stream()
      .filter(nodeTupleKeyEquals(partialKey))
      .map(NodeTuple::getValueNode)
      .findFirst()
      .map(toExistingMappingNode(partialKey))
      .orElseGet(() -> {
        MappingNode newParentConfiguration = newMappingNode();

        parentMappingNode.getValue().add(new NodeTuple(buildScalarNode(partialKey), newParentConfiguration));

        return newParentConfiguration;
      });
  }

  private Function toExistingMappingNode(String partialKey) {
    return valueNode -> {
      if (!(valueNode instanceof MappingNode)) {
        throw GeneratorException.technicalError("Error updating Yaml properties: can't define a subproperty of %s ".formatted(partialKey));
      }

      return (MappingNode) valueNode;
    };
  }

  private MappingNode newMappingNode() {
    return new MappingNode(Tag.MAP, new ArrayList<>(), FlowStyle.AUTO);
  }

  private static Predicate nodeTupleKeyEquals(String partialKey) {
    return nodeTuple -> ((ScalarNode) nodeTuple.getKeyNode()).getValue().equals(partialKey);
  }

  private Node buildScalarNode(Object value) {
    Tag tag = buildTag(value);

    return new ScalarNode(tag, value.toString(), null, null, DumperOptions.ScalarStyle.PLAIN);
  }

  private Tag buildTag(Object value) {
    return TAG_BY_JAVA_TYPE.getOrDefault(value.getClass(), Tag.STR);
  }

  private static List extractKeysParts(PropertyKey key) {
    return Arrays.stream(key.get().split("\\.(?![^'\\[]*])")).map(subKey -> subKey.replace("'[", "[").replace("]'", "]")).toList();
  }

  private MappingNode loadConfiguration(File yamlFile) throws IOException {
    if (!yamlFile.exists()) {
      return new MappingNode(Tag.MAP, new ArrayList<>(), FlowStyle.AUTO);
    }

    try (FileReader reader = new FileReader(yamlFile)) {
      return (MappingNode) yaml.compose(reader);
    }
  }

  private void saveConfiguration(Node actualConfiguration) throws IOException {
    Files.createDirectories(file.getParent());
    Writer writer = new FileWriter(file.toFile());
    yaml.serialize(actualConfiguration, writer);
  }

  private Yaml createYaml() {
    LoaderOptions loaderOptions = new LoaderOptions();
    loaderOptions.setProcessComments(true);

    DumperOptions dumperOptions = new DumperOptions();
    dumperOptions.setProcessComments(true);
    dumperOptions.setIndent(indentation.spacesCount());
    dumperOptions.setPrettyFlow(true);
    dumperOptions.setDefaultFlowStyle(FlowStyle.BLOCK);
    dumperOptions.setSplitLines(true);

    return new Yaml(new Constructor(loaderOptions), new Representer(dumperOptions), dumperOptions, loaderOptions);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy