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

io.vertx.docgen.processor.impl.Helper Maven / Gradle / Ivy

The newest version!
package io.vertx.docgen.processor.impl;

import com.sun.source.tree.CompilationUnitTree;
import com.sun.source.util.DocTrees;
import io.vertx.docgen.Source;

import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.TypeParameterElement;
import javax.lang.model.type.ArrayType;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.ExecutableType;
import javax.lang.model.type.PrimitiveType;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.type.TypeVariable;
import javax.lang.model.util.Elements;
import javax.lang.model.util.Types;
import java.io.IOException;
import java.io.Reader;
import java.lang.reflect.UndeclaredThrowableException;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @author Julien Viet
 */
class Helper {

  static final Pattern LANG_PATTERN = Pattern.compile("(\\\\)?\\$lang");

  final Types typeUtils;
  final Elements elementUtils;
  final DocTrees docTrees;

  public Helper(ProcessingEnvironment env) {
    typeUtils = env.getTypeUtils();
    elementUtils = env.getElementUtils();
    docTrees = DocTrees.instance(env);
  }

  boolean matchesConstructor(Element elt, String memberName, Predicate parametersMatcher) {
    if (elt.getKind() == ElementKind.CONSTRUCTOR) {
      ExecutableElement constructorElt = (ExecutableElement) elt;
      TypeElement typeElt = (TypeElement) constructorElt.getEnclosingElement();
      return typeElt.getSimpleName().toString().equals(memberName) && parametersMatcher.test(constructorElt);
    }
    return false;
  }

  boolean matchesMethod(Element elt, String memberName, Predicate parametersMatcher) {
    if (elt.getKind() == ElementKind.METHOD) {
      ExecutableElement methodElt = (ExecutableElement) elt;
      return methodElt.getSimpleName().toString().equals(memberName) && parametersMatcher.test(methodElt);
    }
    return false;
  }

  boolean matchesFieldOrEnumConstant(Element elt, String memberName) {
    ElementKind kind = elt.getKind();
    return (kind == ElementKind.FIELD || kind == ElementKind.ENUM_CONSTANT) && elt.getSimpleName().toString().equals(memberName);
  }

  private static final Pattern P = Pattern.compile("#(\\p{javaJavaIdentifierStart}(?:\\p{javaJavaIdentifierPart})*)(?:\\((.*)\\))?$");

  public Element resolveLink(String signature) {
    Matcher signatureMatcher = P.matcher(signature);
    if (signatureMatcher.find()) {
      String memberName = signatureMatcher.group(1);
      String typeName = signature.substring(0, signatureMatcher.start());
      TypeElement typeElt = elementUtils.getTypeElement(typeName);
      if (typeElt == null) {
        return null;
      }
      Predicate memberMatcher;
      if (signatureMatcher.group(2) != null) {
        String t = signatureMatcher.group(2).trim();
        Predicate parametersMatcher;
        if (t.length() == 0) {
          parametersMatcher = exeElt -> exeElt.getParameters().isEmpty();
        } else {
          parametersMatcher = parametersMatcher(t.split("\\s*,\\s*"));
        }
        memberMatcher = elt -> matchesConstructor(elt, memberName, parametersMatcher) || matchesMethod(elt, memberName, parametersMatcher);
      } else {
        memberMatcher = elt -> matchesConstructor(elt, memberName, exeElt -> true) ||
            matchesMethod(elt, memberName, exeElt -> true) ||
            matchesFieldOrEnumConstant(elt, memberName);
      }
      // The order of kinds is important
      for (ElementKind kind : Arrays.asList(ElementKind.FIELD, ElementKind.ENUM_CONSTANT, ElementKind.CONSTRUCTOR, ElementKind.METHOD)) {
        for (Element memberElt : elementUtils.getAllMembers(typeElt)) {
          if(memberElt.getKind() == kind && memberMatcher.test(memberElt)) {
            return memberElt;
          }
        }
      }
      return null;
    } else {
      Element elt = elementUtils.getTypeElement(signature);
      if (elt == null) {
        elt = elementUtils.getPackageElement(signature);
      }
      return elt;
    }
  }

  /**
   * Return a matcher for parameters, given the parameter type signature of an executable element. The parameter signature
   * is a list of parameter types formatted as a signature, i.e all types are raw, or primitive, or arrays. Unqualified
   * types are resolved against the import of the specified {@code compilationUnitTree} argument.
   *
   * @param parameterSignature the parameter type names
   * @return the matcher
   */
  Predicate parametersMatcher(String[] parameterSignature) {
    return exeElt -> {
      if (exeElt.getParameters().size() == parameterSignature.length) {
        TypeMirror tm2  = exeElt.asType();
        ExecutableType tm3  = (ExecutableType) typeUtils.erasure(tm2);
        for (int j = 0;j < parameterSignature.length;j++) {
          String t1 = toString(tm3.getParameterTypes().get(j));
          String t2 = parameterSignature[j];
          if (t2.indexOf('.') == -1) {
            t1 = t1.substring(t1.lastIndexOf('.') + 1);
          } else if (t1.indexOf('.') == -1) {
            t2 = t2.substring(t2.lastIndexOf('.') + 1);
          }
          if (!t1.equals(t2)) {
            return false;
          }
        }
        return true;
      } else {
        return false;
      }
    };
  }

  /**
   * Check the element is an example or not.
   *
   * @param elt the elt to check
   * @return true when the checked element is an example
   */
  boolean isExample(Element elt) {
    return elt.getAnnotation(Source.class) != null || elt.getEnclosingElement() != null && isExample(elt.getEnclosingElement());
  }

  /**
   * Check the element is an example or not.
   *
   * @param elt the elt to check
   * @return true when the checked element is an example
   */
  boolean hasToBeTranslated(Element elt) {
    // Find the closest source annotation
    // We are sure to have at least one as this method **must** be called after isExample.
    Element current = elt;
    Source source = current.getAnnotation(Source.class);
    while (source == null) {
      current = current.getEnclosingElement();
      source = current.getAnnotation(Source.class);
    }

    return source.translate();
  }

  /**
   * Read the source code of the provided element, this returns the source of the entire related compilation unit.
   *
   * @param elt the element to load
   * @return the source
   */
  String readSource(Element elt) {
    CompilationUnitTree unit = docTrees.getPath(elt).getCompilationUnit();
    StringBuilder source = new StringBuilder();
    try(Reader reader = unit.getSourceFile().openReader(true)) {
      char[] buffer = new char[256];
      while (true) {
        int len = reader.read(buffer);
        if (len == -1) {
          break;
        }
        source.append(buffer, 0, len);
      }
      return source.toString();
    } catch (IOException e) {
      throw new DocGenException(elt, "Could not read source code of element " + elt);
    }
  }

  /**
   * Compute the string representation of a type mirror.
   *
   * @param mirror the type mirror
   * @return the string representation
   */
  static String toString(TypeMirror mirror) {
    StringBuilder buffer = new StringBuilder();
    toString(mirror, buffer);
    return buffer.toString();
  }

  /**
   * Compute the string representation of a type mirror.
   *
   * @param mirror the type mirror
   * @param buffer the buffer appended with the string representation
   */
  static void toString(TypeMirror mirror, StringBuilder buffer) {
    if (mirror instanceof DeclaredType) {
      DeclaredType dt = (DeclaredType) mirror;
      TypeElement elt = (TypeElement) dt.asElement();
      buffer.append(elt.getQualifiedName().toString());
      List args = dt.getTypeArguments();
      if (args.size() > 0) {
        buffer.append("<");
        for (int i = 0;i < args.size();i++) {
          if (i > 0) {
            buffer.append(",");
          }
          toString(args.get(i), buffer);
        }
        buffer.append(">");
      }
    } else if (mirror instanceof PrimitiveType) {
      PrimitiveType pm = (PrimitiveType) mirror;
      buffer.append(pm.getKind().name().toLowerCase());
    } else if (mirror instanceof javax.lang.model.type.WildcardType) {
      javax.lang.model.type.WildcardType wt = (javax.lang.model.type.WildcardType) mirror;
      buffer.append("?");
      if (wt.getSuperBound() != null) {
        buffer.append(" super ");
        toString(wt.getSuperBound(), buffer);
      } else if (wt.getExtendsBound() != null) {
        buffer.append(" extends ");
        toString(wt.getExtendsBound(), buffer);
      }
    } else if (mirror instanceof javax.lang.model.type.TypeVariable) {
      javax.lang.model.type.TypeVariable tv = (TypeVariable) mirror;
      TypeParameterElement elt = (TypeParameterElement) tv.asElement();
      buffer.append(elt.getSimpleName().toString());
      if (tv.getUpperBound() != null && !tv.getUpperBound().toString().equals("java.lang.Object")) {
        buffer.append(" extends ");
        toString(tv.getUpperBound(), buffer);
      } else if (tv.getLowerBound() != null && tv.getLowerBound().getKind() != TypeKind.NULL) {
        buffer.append(" super ");
        toString(tv.getUpperBound(), buffer);
      }
    } else if (mirror instanceof javax.lang.model.type.ArrayType) {
      javax.lang.model.type.ArrayType at = (ArrayType) mirror;
      toString(at.getComponentType(), buffer);
      buffer.append("[]");
    } else {
      throw new UnsupportedOperationException("todo " + mirror + " " + mirror.getKind());
    }
  }

  public void filterLang(CharSequence csq, String lang, Appendable dst) {
    try {
      Matcher matcher = Helper.LANG_PATTERN.matcher(csq);
      int prev = 0;
      while (matcher.find()) {
        dst.append(csq, prev, matcher.start());
        if (matcher.group(1) != null) {
          // \$lang
          dst.append("$lang");
        } else {
          dst.append(lang);
        }
        prev = matcher.end();
      }
      dst.append(csq, prev, csq.length());
    } catch (IOException e) {
      throw new UndeclaredThrowableException(e);
    }
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy