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

io.github.microcks.util.AbstractJsonRepositoryImporter Maven / Gradle / Ivy

The newest version!
/*
 * Copyright The Microcks Authors.
 *
 * 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.github.microcks.util;

import io.github.microcks.domain.Resource;
import io.github.microcks.domain.ResourceType;
import io.github.microcks.domain.Service;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.JsonNodeType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

/**
 * An abstract class that may be used as a base for implementing MockRepositoryImporter that are using JSON/YAML file as
 * a repository. It provides utility methods for handling references and loading of external resources, resolution of
 * JSON pointers and so on...
 * @author laurent
 */
public abstract class AbstractJsonRepositoryImporter {

   /** A simple logger for diagnostic messages. */
   private static final Logger log = LoggerFactory.getLogger(AbstractJsonRepositoryImporter.class);

   protected Boolean isYaml;
   protected JsonNode rootSpecification;
   protected String rootSpecificationContent;
   protected ReferenceResolver referenceResolver;
   protected List externalResources = new ArrayList<>();
   protected Map externalResourcesContent = new HashMap<>();

   /**
    * Build a new importer using the path of specification file and an optional reference resolver.
    * @param specificationFilePath The path to local JSON spec file
    * @param referenceResolver     An optional resolver for references present into the OpenAPI file
    * @throws IOException if JSON spec file cannot be found or read.
    */
   protected AbstractJsonRepositoryImporter(String specificationFilePath, ReferenceResolver referenceResolver)
         throws IOException {
      this.referenceResolver = referenceResolver;
      BufferedReader reader = null;
      try {
         // Analyse first lines of file content to guess repository type.
         String line = null;
         reader = Files.newBufferedReader(new File(specificationFilePath).toPath(), StandardCharsets.UTF_8);
         while ((line = reader.readLine()) != null && isYaml == null) {
            line = line.trim();
            // Check is we start with json object or array definition.
            if (line.startsWith("{") || line.startsWith("[")) {
               isYaml = false;
            } else if (line.startsWith("---") || line.startsWith("-") || line.startsWith("openapi: ")
                  || line.startsWith("asyncapi: ")) {
               isYaml = true;
            }
         }

         // Read spec bytes.
         byte[] bytes = Files.readAllBytes(Paths.get(specificationFilePath));
         rootSpecificationContent = new String(bytes, StandardCharsets.UTF_8);

         // Convert them to Node using Jackson object mapper.
         ObjectMapper mapper = getObjectMapper(isYaml);
         rootSpecification = mapper.readTree(rootSpecificationContent.getBytes(StandardCharsets.UTF_8));
      } catch (Exception e) {
         log.error("Exception while parsing JSON specification file " + specificationFilePath, e);
         throw new IOException("JSON spec file parsing error");
      } finally {
         if (reader != null) {
            reader.close();
         }
      }
   }

   /**
    * Given a service, initialize all external references that may be found via `$ref` nodes pointing to absolute or
    * relative pointers. This has the side effect to initialize the `externalResources` member.
    * @param service The main service for naming discovered resources.
    */
   protected void initializeReferencedResources(Service service) throws MockRepositoryImportException {
      if (referenceResolver != null) {
         String rootBaseUrl = referenceResolver.getBaseRepositoryUrl();

         // Keep track and collect reference resources, using absolute URL as key.
         Map referenceResources = new HashMap<>();

         // We need to create a temporary root resource to initiate the recursive resolution.
         Resource rootResource = new Resource();
         rootResource.setName(service.getName() + "-" + service.getVersion());
         rootResource.setContent(rootSpecificationContent);
         referenceResources.put(rootBaseUrl, rootResource);

         // Seek, resolve and build reference resources.
         resolveExternalReferences(service, referenceResources, rootBaseUrl, "", rootSpecification);

         // Secondly: update root specification content with new reference names and
         // Refresh the root specification node with re-normalized references.
         rootSpecificationContent = rootResource.getContent();
         try {
            ObjectMapper mapper = getObjectMapper(isYaml);
            rootSpecification = mapper.readTree(rootSpecificationContent.getBytes(StandardCharsets.UTF_8));
         } catch (Exception e) {
            log.error("Exception while parsing re-normalized JSON specification file", e);
            throw new MockRepositoryImportException("Exception while parsing re-normalized JSON specification file");
         }

         // Finally try to clean up resolved references and associated resources (files)
         referenceResolver.cleanResolvedReferences();
      }
   }

   /**
    * Recursive method for browsing resource spec, finding $ref, downloading content and accumulating result in
    * referenceResources.
    */
   private void resolveExternalReferences(Service service, Map referenceResources,
         String baseRepositoryUrl, String baseContext, JsonNode resourceSpecification) {
      Set references = findAllExternalRefs(resourceSpecification);
      Resource currentResource = referenceResources.get(baseRepositoryUrl);

      for (String ref : references) {
         referenceResolver.setBaseRepositoryUrl(baseRepositoryUrl);
         String refUrl = referenceResolver.getReferenceURL(ref);

         Resource referenceResource = referenceResources.get(refUrl);
         if (referenceResource == null) {
            try {
               // Extract content using resolver.
               String content = referenceResolver.getReferenceContent(ref, StandardCharsets.UTF_8);

               // Build resource name from short name.
               String resourceName = ref.substring(ref.lastIndexOf('/') + 1);
               String referencePath = ref.contains("/") ? ref.substring(0, ref.lastIndexOf('/')) : ".";
               // Introduce a context tracker if it's a relative dependency.
               String referenceContext = buildContext(baseContext, referencePath);
               if (!ref.startsWith("http")) {
                  resourceName = IdBuilder.buildResourceFullName(service, resourceName, referenceContext);
               } else {
                  resourceName = IdBuilder.buildResourceFullName(service, resourceName);
               }

               // Build a new resource from content. Use the escaped operation path.
               referenceResource = new Resource();
               referenceResource.setName(resourceName);
               referenceResource.setPath(ref);
               referenceResource.setContent(content);
               referenceResource.setType(guessResourceType(ref, content));

               // Keep track of this newly created resource.
               referenceResources.put(refUrl, referenceResource);
               externalResources.add(referenceResource);

               // Now go down the resource content and resolve its embedded references.
               // Also update the references bases to track the root url for this ref in order
               ObjectMapper mapper = getObjectMapper(!ref.endsWith(".json"));
               JsonNode refResourceSpecification = mapper.readTree(content);
               resolveExternalReferences(service, referenceResources, refUrl, referenceContext,
                     refResourceSpecification);
            } catch (IOException ioe) {
               log.error("IOException while trying to resolve reference {}", ref, ioe);
               log.info("Ignoring the reference {} cause it could not be resolved", ref);
            }
         }

         if (!ref.startsWith("http") && referenceResource != null) {
            // If a relative resource, replace with new name in resource content.
            String refNewName = referenceResources.get(refUrl).getName();
            String normalizedContent = currentResource.getContent().replace(ref,
                  URLEncoder.encode(refNewName, StandardCharsets.UTF_8));
            currentResource.setContent(normalizedContent);
         }
      }
   }

   /**
    * Build a context for resource name in order to avoid name collisions (resource having same short name in different
    * folders.
    */
   private String buildContext(String baseContext, String referencePath) {
      // Treat the obvious "." case.
      if (".".equals(referencePath)) {
         return baseContext;
      }
      // Else recompose a path to append.
      String pathToAppend = referencePath;
      while (pathToAppend.startsWith("../")) {
         if (baseContext.contains("/")) {
            baseContext = baseContext.substring(0, baseContext.lastIndexOf("/"));
         }
         pathToAppend = pathToAppend.substring(3);
      }
      if (pathToAppend.startsWith("./")) {
         pathToAppend = pathToAppend.substring(2);
      }
      if (pathToAppend.startsWith("/")) {
         pathToAppend = pathToAppend.substring(1);
      }
      String result = baseContext + "/" + pathToAppend;
      return result.startsWith("/") ? result.substring(1) : result;
   }

   /** Follow the $ref if we have one. Otherwise, return given node. */
   protected JsonNode followRefIfAny(JsonNode referencableNode) {
      if (referencableNode.has("$ref")) {
         String ref = referencableNode.path("$ref").asText();
         return getNodeForRef(ref);
      }
      return referencableNode;
   }

   /** Get the string representation of a node in spec. */
   protected String getValueString(JsonNode valueNode) {
      // Get string representation if array or object.
      if (valueNode.getNodeType() == JsonNodeType.ARRAY || valueNode.getNodeType() == JsonNodeType.OBJECT) {
         return valueNode.toString();
      }
      // Else get raw representation.
      return valueNode.asText();
   }

   /** Get appropriate Yaml or Json object mapper. */
   protected ObjectMapper getObjectMapper(boolean isYaml) {
      return isYaml ? ObjectMapperFactory.getYamlObjectMapper() : ObjectMapperFactory.getJsonObjectMapper();
   }

   /** Browse Json node to extract references and store them into externalRefs. */
   private Set findAllExternalRefs(JsonNode node) {
      Set externalRefs = new HashSet<>();
      // If node has a $ref child, it's a stop condition.
      if (node.has("$ref")) {
         String ref = node.path("$ref").asText();
         if (!ref.startsWith("#")) {
            // Our ref could be used for examples and be something like './weather-examples.json#/0'.
            String filePath = ref;
            if (ref.contains("#/")) {
               filePath = ref.substring(0, ref.indexOf("#/"));
            }
            externalRefs.add(filePath);
         }
      } else {
         // Iterate on all other children.
         Iterator children = node.elements();
         while (children.hasNext()) {
            externalRefs.addAll(findAllExternalRefs(children.next()));
         }
      }
      return externalRefs;
   }

   /** Get the JsonNode for reference within the specification. */
   private JsonNode getNodeForRef(String reference) {
      if (reference.startsWith("#/")) {
         return rootSpecification.at(reference.substring(1));
      }
      return getNodeForExternalRef(reference);
   }

   /** Get the JsonNode for reference that is localed in external resource. */
   private JsonNode getNodeForExternalRef(String externalReference) {
      String path = externalReference;

      // We may have a Json pointer to a specific place in external reference.
      String pointerInFile = null;
      if (externalReference.indexOf("#/") != -1) {
         path = externalReference.substring(0, externalReference.indexOf("#/"));
         pointerInFile = externalReference.substring(externalReference.indexOf("#/"));
      }

      for (Resource resource : externalResources) {
         // Path direct equality is for absolute ref ("http://raw.githubusercontent.com/...")
         // Path equality with resource name if for relative refs that have been re-normalized ("Service name+Service version+...))
         if (path.equals(resource.getPath())
               || path.equals(URLEncoder.encode(resource.getName(), StandardCharsets.UTF_8))) {
            JsonNode resourceNode = externalResourcesContent.computeIfAbsent(resource, k -> {
               try {
                  return ObjectMapperFactory.getYamlObjectMapper().readTree(resource.getContent());
               } catch (JsonProcessingException e) {
                  throw new JsonRepositoryParsingException("Get a JSON processing exception on " + externalReference,
                        e);
               }
            });
            if (pointerInFile != null) {
               return resourceNode.at(pointerInFile.substring(1));
            }
            return resourceNode;
         }
      }
      log.warn("Found no resource for reference {}", externalReference);
      return null;
   }

   /** Try to guess resource type from a reference name and its content. */
   private ResourceType guessResourceType(String ref, String content) {
      if (ref.endsWith(".avsc")) {
         return ResourceType.AVRO_SCHEMA;
      } else if (ref.endsWith(".proto")) {
         return ResourceType.PROTOBUF_SCHEMA;
      } else if (content.contains("$schema") || content.contains("properties:")
            || content.contains("\"properties\":")) {
         return ResourceType.JSON_SCHEMA;
      }
      return ResourceType.JSON_FRAGMENT;
   }

   /** Custom runtime exception for Json repository parsing errors. */
   public class JsonRepositoryParsingException extends RuntimeException {
      public JsonRepositoryParsingException(String message) {
         super(message);
      }

      public JsonRepositoryParsingException(String message, Throwable cause) {
         super(message, cause);
      }
   }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy