From c0c5ffc7a2b2d2e0941c189aa2fe6c7f886cdc57 Mon Sep 17 00:00:00 2001 From: Alan Zimmer <48699787+alzimmermsft@users.noreply.github.com> Date: Wed, 17 Apr 2024 12:31:19 -0400 Subject: [PATCH 1/5] Add method to handle indenting and unindenting a block of tokens --- .../processor/analysers/JavaASTAnalyser.java | 532 +++++++++--------- 1 file changed, 273 insertions(+), 259 deletions(-) diff --git a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java index 89dc7924787..dfc0ba1bfe0 100644 --- a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java +++ b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java @@ -144,7 +144,7 @@ public class JavaASTAnalyser implements Analyser { private final Diagnostics diagnostic; - private int indent = 0; + private int indentLevel = 0; public JavaASTAnalyser(APIListing apiListing) { this.apiListing = apiListing; @@ -271,13 +271,9 @@ private void processPackage(String packageName, List scanClasses) { addToken(packageToken, SPACE); addToken(new Token(PUNCTUATION, "{"), NEWLINE); - indent(); - - scanClasses.stream() + indentedBlock(() -> scanClasses.stream() .sorted(Comparator.comparing(s -> s.primaryTypeName)) - .forEach(this::processSingleFile); - - unindent(); + .forEach(this::processSingleFile)); addToken(new Token(PUNCTUATION, "}"), NEWLINE); } @@ -302,96 +298,103 @@ private void tokeniseMavenPom(Pom mavenPom) { addToken(makeWhitespace()); addToken(new Token(KEYWORD, "maven", MAVEN_KEY), SPACE); addToken(new Token(PUNCTUATION, "{"), NEWLINE); - indent(); - - addToken(new Token(SKIP_DIFF_START)); - // parent - String gavStr = mavenPom.getParent().getGroupId() + ":" + mavenPom.getParent().getArtifactId() + ":" + mavenPom.getParent().getVersion(); - tokeniseKeyValue("parent", gavStr, ""); - // properties - gavStr = mavenPom.getGroupId() + ":" + mavenPom.getArtifactId() + ":" + mavenPom.getVersion(); - tokeniseKeyValue("properties", gavStr, ""); + indentedBlock(() -> { + addToken(new Token(SKIP_DIFF_START)); + // parent + String gavStr = mavenPom.getParent().getGroupId() + ":" + mavenPom.getParent().getArtifactId() + ":" + + mavenPom.getParent().getVersion(); + tokeniseKeyValue("parent", gavStr, ""); + + // properties + gavStr = mavenPom.getGroupId() + ":" + mavenPom.getArtifactId() + ":" + mavenPom.getVersion(); + tokeniseKeyValue("properties", gavStr, ""); + + // configuration + boolean showJacoco = mavenPom.getJacocoMinLineCoverage() != null + && mavenPom.getJacocoMinBranchCoverage() != null; + boolean showCheckStyle = mavenPom.getCheckstyleExcludes() != null && !mavenPom.getCheckstyleExcludes() + .isEmpty(); + + if (showJacoco || showCheckStyle) { + addToken(INDENT, new Token(KEYWORD, "configuration"), SPACE); + addToken(new Token(PUNCTUATION, "{"), NEWLINE); - // configuration - boolean showJacoco = mavenPom.getJacocoMinLineCoverage() != null && - mavenPom.getJacocoMinBranchCoverage() != null; - boolean showCheckStyle = mavenPom.getCheckstyleExcludes() != null && - !mavenPom.getCheckstyleExcludes().isEmpty(); + indentedBlock(() -> { + if (showCheckStyle) { + tokeniseKeyValue("checkstyle-excludes", mavenPom.getCheckstyleExcludes(), ""); + } + if (showJacoco) { + addToken(INDENT, new Token(KEYWORD, "jacoco"), SPACE); + addToken(new Token(PUNCTUATION, "{"), NEWLINE); + indentedBlock(() -> { + tokeniseKeyValue("min-line-coverage", mavenPom.getJacocoMinLineCoverage(), "jacoco"); + tokeniseKeyValue("min-branch-coverage", mavenPom.getJacocoMinBranchCoverage(), "jacoco"); + }); + addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); + } + }); - if (showJacoco || showCheckStyle) { - addToken(INDENT, new Token(KEYWORD, "configuration"), SPACE); - addToken(new Token(PUNCTUATION, "{"), NEWLINE); - indent(); - if (showCheckStyle) { - tokeniseKeyValue("checkstyle-excludes", mavenPom.getCheckstyleExcludes(), ""); - } - if (showJacoco) { - addToken(INDENT, new Token(KEYWORD, "jacoco"), SPACE); - addToken(new Token(PUNCTUATION, "{"), NEWLINE); - indent(); - tokeniseKeyValue("min-line-coverage", mavenPom.getJacocoMinLineCoverage(), "jacoco"); - tokeniseKeyValue("min-branch-coverage", mavenPom.getJacocoMinBranchCoverage(), "jacoco"); - unindent(); addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); } - unindent(); - addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); - } - // Maven name - tokeniseKeyValue("name", mavenPom.getName(), ""); + // Maven name + tokeniseKeyValue("name", mavenPom.getName(), ""); - // Maven description - tokeniseKeyValue("description", mavenPom.getDescription(), ""); + // Maven description + tokeniseKeyValue("description", mavenPom.getDescription(), ""); - // dependencies - addToken(INDENT, new Token(KEYWORD, "dependencies"), SPACE); - addToken(new Token(PUNCTUATION, "{"), NEWLINE); + // dependencies + addToken(INDENT, new Token(KEYWORD, "dependencies"), SPACE); + addToken(new Token(PUNCTUATION, "{"), NEWLINE); - indent(); - mavenPom.getDependencies().stream().collect(Collectors.groupingBy(Dependency::getScope)).forEach((k, v) -> { - if ("test".equals(k)) { - // we don't care to present test scope dependencies - return; - } - String scope = k.isEmpty() ? "compile" : k; + indentedBlock(() -> { + mavenPom.getDependencies() + .stream() + .collect(Collectors.groupingBy(Dependency::getScope)) + .forEach((k, v) -> { + if ("test".equals(k)) { + // we don't care to present test scope dependencies + return; + } + String scope = k.isEmpty() ? "compile" : k; - addToken(makeWhitespace()); - addToken(new Token(COMMENT, "// " + scope + " scope"), NEWLINE); + addToken(makeWhitespace()); + addToken(new Token(COMMENT, "// " + scope + " scope"), NEWLINE); - for (int i = 0; i < v.size(); i++) { - Dependency d = v.get(i); - addToken(makeWhitespace()); - String gav = d.getGroupId() + ":" + d.getArtifactId() + ":" + d.getVersion(); - addToken(new Token(TEXT, gav, gav)); + for (int i = 0; i < v.size(); i++) { + Dependency d = v.get(i); + addToken(makeWhitespace()); + String gav = d.getGroupId() + ":" + d.getArtifactId() + ":" + d.getVersion(); + addToken(new Token(TEXT, gav, gav)); - if (i < v.size() - 1) { - addNewLine(); - } - } + if (i < v.size() - 1) { + addNewLine(); + } + } - addNewLine(); + addNewLine(); + }); + }); + + addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); + addToken(new Token(SKIP_DIFF_END)); + + // allowed dependencies (in maven-enforcer) + // if (!mavenPom.getAllowedDependencies().isEmpty()) { + // addToken(INDENT, new Token(KEYWORD, "allowed-dependencies"), SPACE); + // addToken(new Token(PUNCTUATION, "{"), NEWLINE); + // indent(); + // mavenPom.getAllowedDependencies().stream().forEach(value -> { + // addToken(INDENT, new Token(TEXT, value, value), NEWLINE); + // }); + // unindent(); + // addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); + // } + + // close maven }); - unindent(); - addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); - addToken(new Token(SKIP_DIFF_END)); - - // allowed dependencies (in maven-enforcer) -// if (!mavenPom.getAllowedDependencies().isEmpty()) { -// addToken(INDENT, new Token(KEYWORD, "allowed-dependencies"), SPACE); -// addToken(new Token(PUNCTUATION, "{"), NEWLINE); -// indent(); -// mavenPom.getAllowedDependencies().stream().forEach(value -> { -// addToken(INDENT, new Token(TEXT, value, value), NEWLINE); -// }); -// unindent(); -// addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); -// } - - // close maven - unindent(); addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); } @@ -505,9 +508,7 @@ private void visitClassOrInterfaceOrEnumDeclaration(TypeDeclaration typeDecla if (typeDeclaration.getMembers().isEmpty()) { // we have an empty interface declaration, it is probably a marker interface and we will leave a // comment to that effect - indent(); - addComment("// This interface is does not declare any API."); - unindent(); + indentedBlock(() -> addComment("// This interface is does not declare any API.")); } } @@ -540,62 +541,66 @@ private void visitModuleDeclaration(ModuleDeclaration moduleDeclaration) { }; moduleDeclaration.getDirectives().forEach(moduleDirective -> { - indent(); - addToken(makeWhitespace()); + indentedBlock(() -> { + addToken(makeWhitespace()); - moduleDirective.ifModuleRequiresStmt(d -> { - addToken(new Token(KEYWORD, "requires"), SPACE); + moduleDirective.ifModuleRequiresStmt(d -> { + addToken(new Token(KEYWORD, "requires"), SPACE); - if (d.hasModifier(Modifier.Keyword.STATIC)) { - addToken(new Token(KEYWORD, "static"), SPACE); - } + if (d.hasModifier(Modifier.Keyword.STATIC)) { + addToken(new Token(KEYWORD, "static"), SPACE); + } - if (d.isTransitive()) { - addToken(new Token(KEYWORD, "transitive"), SPACE); - } + if (d.isTransitive()) { + addToken(new Token(KEYWORD, "transitive"), SPACE); + } - addToken(new Token(TYPE_NAME, d.getNameAsString(), makeId(MODULE_INFO_KEY + "-requires-" + d.getNameAsString()))); - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - }); + addToken(new Token(TYPE_NAME, d.getNameAsString(), + makeId(MODULE_INFO_KEY + "-requires-" + d.getNameAsString()))); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + }); - moduleDirective.ifModuleExportsStmt(d -> { - addToken(new Token(KEYWORD, "exports"), SPACE); - addToken(new Token(TYPE_NAME, d.getNameAsString(), makeId(MODULE_INFO_KEY + "-exports-" + d.getNameAsString()))); - conditionalExportsToOrOpensToConsumer.accept(d.getModuleNames()); - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - }); + moduleDirective.ifModuleExportsStmt(d -> { + addToken(new Token(KEYWORD, "exports"), SPACE); + addToken(new Token(TYPE_NAME, d.getNameAsString(), + makeId(MODULE_INFO_KEY + "-exports-" + d.getNameAsString()))); + conditionalExportsToOrOpensToConsumer.accept(d.getModuleNames()); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + }); - moduleDirective.ifModuleOpensStmt(d -> { - addToken(new Token(KEYWORD, "opens"), SPACE); - addToken(new Token(TYPE_NAME, d.getNameAsString(), makeId(MODULE_INFO_KEY + "-opens-" + d.getNameAsString()))); - conditionalExportsToOrOpensToConsumer.accept(d.getModuleNames()); - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - }); + moduleDirective.ifModuleOpensStmt(d -> { + addToken(new Token(KEYWORD, "opens"), SPACE); + addToken(new Token(TYPE_NAME, d.getNameAsString(), + makeId(MODULE_INFO_KEY + "-opens-" + d.getNameAsString()))); + conditionalExportsToOrOpensToConsumer.accept(d.getModuleNames()); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + }); - moduleDirective.ifModuleUsesStmt(d -> { - addToken(new Token(KEYWORD, "uses"), SPACE); - addToken(new Token(TYPE_NAME, d.getNameAsString(), makeId(MODULE_INFO_KEY + "-uses-" + d.getNameAsString()))); - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - }); + moduleDirective.ifModuleUsesStmt(d -> { + addToken(new Token(KEYWORD, "uses"), SPACE); + addToken(new Token(TYPE_NAME, d.getNameAsString(), + makeId(MODULE_INFO_KEY + "-uses-" + d.getNameAsString()))); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + }); - moduleDirective.ifModuleProvidesStmt(d -> { - addToken(new Token(KEYWORD, "provides"), SPACE); - addToken(new Token(TYPE_NAME, d.getNameAsString(), makeId(MODULE_INFO_KEY + "-provides-" + d.getNameAsString())), SPACE); - addToken(new Token(KEYWORD, "with"), SPACE); + moduleDirective.ifModuleProvidesStmt(d -> { + addToken(new Token(KEYWORD, "provides"), SPACE); + addToken(new Token(TYPE_NAME, d.getNameAsString(), + makeId(MODULE_INFO_KEY + "-provides-" + d.getNameAsString())), SPACE); + addToken(new Token(KEYWORD, "with"), SPACE); - NodeList names = d.getWith(); - for (int i = 0; i < names.size(); i++) { - addToken(new Token(TYPE_NAME, names.get(i).toString())); + NodeList names = d.getWith(); + for (int i = 0; i < names.size(); i++) { + addToken(new Token(TYPE_NAME, names.get(i).toString())); - if (i < names.size() - 1) { - addToken(new Token(PUNCTUATION, ","), SPACE); + if (i < names.size() - 1) { + addToken(new Token(PUNCTUATION, ","), SPACE); + } } - } - addToken(new Token(PUNCTUATION, ";"), NEWLINE); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + }); }); - - unindent(); }); // close module @@ -605,49 +610,48 @@ private void visitModuleDeclaration(ModuleDeclaration moduleDeclaration) { private void getEnumEntries(EnumDeclaration enumDeclaration) { final NodeList enumConstantDeclarations = enumDeclaration.getEntries(); int size = enumConstantDeclarations.size(); - indent(); - AtomicInteger counter = new AtomicInteger(); + indentedBlock(() -> { + AtomicInteger counter = new AtomicInteger(); - enumConstantDeclarations.forEach(enumConstantDeclaration -> { - visitJavaDoc(enumConstantDeclaration); + enumConstantDeclarations.forEach(enumConstantDeclaration -> { + visitJavaDoc(enumConstantDeclaration); - addToken(makeWhitespace()); + addToken(makeWhitespace()); - // annotations - getAnnotations(enumConstantDeclaration, false, false); + // annotations + getAnnotations(enumConstantDeclaration, false, false); - // create a unique id for enum constants by using the fully-qualified constant name - // (package, enum name, and enum constant name) - final String name = enumConstantDeclaration.getNameAsString(); - final String definitionId = makeId(enumConstantDeclaration); - final boolean isDeprecated = enumConstantDeclaration.isAnnotationPresent("Deprecated"); + // create a unique id for enum constants by using the fully-qualified constant name + // (package, enum name, and enum constant name) + final String name = enumConstantDeclaration.getNameAsString(); + final String definitionId = makeId(enumConstantDeclaration); + final boolean isDeprecated = enumConstantDeclaration.isAnnotationPresent("Deprecated"); - if (isDeprecated) { - addToken(new Token(DEPRECATED_RANGE_START)); - } + if (isDeprecated) { + addToken(new Token(DEPRECATED_RANGE_START)); + } - addToken(new Token(MEMBER_NAME, name, definitionId)); + addToken(new Token(MEMBER_NAME, name, definitionId)); - if (isDeprecated) { - addToken(new Token(DEPRECATED_RANGE_END)); - } + if (isDeprecated) { + addToken(new Token(DEPRECATED_RANGE_END)); + } - enumConstantDeclaration.getArguments().forEach(expression -> { - addToken(new Token(PUNCTUATION, "(")); - addToken(new Token(TEXT, expression.toString())); - addToken(new Token(PUNCTUATION, ")")); - }); + enumConstantDeclaration.getArguments().forEach(expression -> { + addToken(new Token(PUNCTUATION, "(")); + addToken(new Token(TEXT, expression.toString())); + addToken(new Token(PUNCTUATION, ")")); + }); - if (counter.getAndIncrement() < size - 1) { - addToken(new Token(PUNCTUATION, ",")); - } else { - addToken(new Token(PUNCTUATION, ";")); - } - addNewLine(); + if (counter.getAndIncrement() < size - 1) { + addToken(new Token(PUNCTUATION, ",")); + } else { + addToken(new Token(PUNCTUATION, ";")); + } + addNewLine(); + }); }); - - unindent(); } private void getTypeDeclaration(TypeDeclaration typeDeclaration) { @@ -781,113 +785,118 @@ private void checkForCrossLanguageDefinitionId(Token typeNameToken, NodeWithSimp } private void tokeniseAnnotationMember(AnnotationDeclaration annotationDeclaration) { - indent(); - // Member methods in the annotation declaration - NodeList> annotationDeclarationMembers = annotationDeclaration.getMembers(); - for (BodyDeclaration bodyDeclaration : annotationDeclarationMembers) { - Optional annotationMemberDeclarationOptional = bodyDeclaration.toAnnotationMemberDeclaration(); - if (!annotationMemberDeclarationOptional.isPresent()) { - continue; - } - final AnnotationMemberDeclaration annotationMemberDeclaration = annotationMemberDeclarationOptional.get(); + indentedBlock(() -> { + // Member methods in the annotation declaration + NodeList> annotationDeclarationMembers = annotationDeclaration.getMembers(); + for (BodyDeclaration bodyDeclaration : annotationDeclarationMembers) { + Optional annotationMemberDeclarationOptional + = bodyDeclaration.toAnnotationMemberDeclaration(); + if (!annotationMemberDeclarationOptional.isPresent()) { + continue; + } + final AnnotationMemberDeclaration annotationMemberDeclaration + = annotationMemberDeclarationOptional.get(); - addToken(makeWhitespace()); - getClassType(annotationMemberDeclaration.getType()); - addToken(new Token(WHITESPACE, " ")); + addToken(makeWhitespace()); + getClassType(annotationMemberDeclaration.getType()); + addToken(new Token(WHITESPACE, " ")); - final String name = annotationMemberDeclaration.getNameAsString(); - final String definitionId = makeId(annotationDeclaration.getFullyQualifiedName().get() + "." + name); + final String name = annotationMemberDeclaration.getNameAsString(); + final String definitionId = makeId( + annotationDeclaration.getFullyQualifiedName().get() + "." + name); - addToken(new Token(MEMBER_NAME, name, definitionId)); - addToken(new Token(PUNCTUATION, "(")); - addToken(new Token(PUNCTUATION, ")")); + addToken(new Token(MEMBER_NAME, name, definitionId)); + addToken(new Token(PUNCTUATION, "(")); + addToken(new Token(PUNCTUATION, ")")); - // default value - final Optional defaultValueOptional = annotationMemberDeclaration.getDefaultValue(); - if (defaultValueOptional.isPresent()) { - addToken(SPACE, new Token(KEYWORD, "default"), SPACE); + // default value + final Optional defaultValueOptional = annotationMemberDeclaration.getDefaultValue(); + if (defaultValueOptional.isPresent()) { + addToken(SPACE, new Token(KEYWORD, "default"), SPACE); - final Expression defaultValueExpr = defaultValueOptional.get(); - final String value = defaultValueExpr.toString(); - addToken(new Token(KEYWORD, value)); - } + final Expression defaultValueExpr = defaultValueOptional.get(); + final String value = defaultValueExpr.toString(); + addToken(new Token(KEYWORD, value)); + } - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - } - unindent(); + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + } + }); } private void tokeniseFields(boolean isInterfaceDeclaration, TypeDeclaration typeDeclaration) { final List fieldDeclarations = typeDeclaration.getFields(); final String fullPathName = typeDeclaration.getFullyQualifiedName().get(); - indent(); - for (FieldDeclaration fieldDeclaration : fieldDeclarations) { - // By default , interface has public abstract methods if there is no access specifier declared - if (isInterfaceDeclaration) { - // no-op - we take all methods in the method - } else if (isPrivateOrPackagePrivate(fieldDeclaration.getAccessSpecifier())) { - // Skip if not public API - continue; - } + indentedBlock(() -> { + for (FieldDeclaration fieldDeclaration : fieldDeclarations) { + // By default , interface has public abstract methods if there is no access specifier declared + if (isInterfaceDeclaration) { + // no-op - we take all methods in the method + } else if (isPrivateOrPackagePrivate(fieldDeclaration.getAccessSpecifier())) { + // Skip if not public API + continue; + } - visitJavaDoc(fieldDeclaration); + visitJavaDoc(fieldDeclaration); - addToken(makeWhitespace()); + addToken(makeWhitespace()); - // Add annotation for field declaration - getAnnotations(fieldDeclaration, false, false); + // Add annotation for field declaration + getAnnotations(fieldDeclaration, false, false); - final NodeList fieldModifiers = fieldDeclaration.getModifiers(); - // public, protected, static, final - for (final Modifier fieldModifier : fieldModifiers) { - addToken(new Token(KEYWORD, fieldModifier.toString())); - } + final NodeList fieldModifiers = fieldDeclaration.getModifiers(); + // public, protected, static, final + for (final Modifier fieldModifier : fieldModifiers) { + addToken(new Token(KEYWORD, fieldModifier.toString())); + } - // field type and name - final NodeList variableDeclarators = fieldDeclaration.getVariables(); + // field type and name + final NodeList variableDeclarators = fieldDeclaration.getVariables(); - if (variableDeclarators.size() > 1) { - getType(fieldDeclaration); + if (variableDeclarators.size() > 1) { + getType(fieldDeclaration); - for (VariableDeclarator variableDeclarator : variableDeclarators) { + for (VariableDeclarator variableDeclarator : variableDeclarators) { + final String name = variableDeclarator.getNameAsString(); + final String definitionId = makeId(fullPathName + "." + variableDeclarator.getName()); + addToken(new Token(MEMBER_NAME, name, definitionId)); + addToken(new Token(PUNCTUATION, ","), SPACE); + } + apiListing.getTokens().remove(apiListing.getTokens().size() - 1); + apiListing.getTokens().remove(apiListing.getTokens().size() - 1); + } else if (variableDeclarators.size() == 1) { + getType(fieldDeclaration); + final VariableDeclarator variableDeclarator = variableDeclarators.get(0); final String name = variableDeclarator.getNameAsString(); final String definitionId = makeId(fullPathName + "." + variableDeclarator.getName()); addToken(new Token(MEMBER_NAME, name, definitionId)); - addToken(new Token(PUNCTUATION, ","), SPACE); - } - apiListing.getTokens().remove(apiListing.getTokens().size() - 1); - apiListing.getTokens().remove(apiListing.getTokens().size() - 1); - } else if (variableDeclarators.size() == 1) { - getType(fieldDeclaration); - final VariableDeclarator variableDeclarator = variableDeclarators.get(0); - final String name = variableDeclarator.getNameAsString(); - final String definitionId = makeId(fullPathName + "." + variableDeclarator.getName()); - addToken(new Token(MEMBER_NAME, name, definitionId)); - final Optional variableDeclaratorOption = variableDeclarator.getInitializer(); - if (variableDeclaratorOption.isPresent()) { - Expression e = variableDeclaratorOption.get(); - if (e.isObjectCreationExpr() && e.asObjectCreationExpr().getAnonymousClassBody().isPresent()) { - // no-op because we don't want to include all of the anonymous inner class details - } else { - addToken(SPACE, new Token(PUNCTUATION, "="), SPACE); - addToken(new Token(TEXT, variableDeclaratorOption.get().toString())); + final Optional variableDeclaratorOption = variableDeclarator.getInitializer(); + if (variableDeclaratorOption.isPresent()) { + Expression e = variableDeclaratorOption.get(); + if (e.isObjectCreationExpr() && e.asObjectCreationExpr() + .getAnonymousClassBody() + .isPresent()) { + // no-op because we don't want to include all of the anonymous inner class details + } else { + addToken(SPACE, new Token(PUNCTUATION, "="), SPACE); + addToken(new Token(TEXT, variableDeclaratorOption.get().toString())); + } } } - } - // close the variable declaration - addToken(new Token(PUNCTUATION, ";"), NEWLINE); - } - unindent(); + // close the variable declaration + addToken(new Token(PUNCTUATION, ";"), NEWLINE); + } + }); } private void tokeniseConstructorsOrMethods(final TypeDeclaration typeDeclaration, final boolean isInterfaceDeclaration, final boolean isConstructor, final List> callableDeclarations) { - indent(); + indentedBlock(() -> { if (isConstructor) { // determining if we are looking at a set of constructors that are all private, indicating that the @@ -900,12 +909,10 @@ private void tokeniseConstructorsOrMethods(final TypeDeclaration typeDeclarat if (isAllPrivateOrPackagePrivate) { if (typeDeclaration.isEnumDeclaration()) { - unindent(); return; } addComment("// This class does not have any public constructors, and is not able to be instantiated using 'new'."); - unindent(); return; } } @@ -981,8 +988,7 @@ private void tokeniseConstructorsOrMethods(final TypeDeclaration typeDeclarat addNewLine(); }); }); - - unindent(); + }); } private int sortMethods(CallableDeclaration c1, CallableDeclaration c2) { @@ -1044,9 +1050,8 @@ private int sortMethods(CallableDeclaration c1, CallableDeclaration c2) { private void tokeniseInnerClasses(Stream> innerTypes) { innerTypes.forEach(innerType -> { if (innerType.isEnumDeclaration() || innerType.isClassOrInterfaceDeclaration()) { - indent(); - new ClassOrInterfaceVisitor(parentNav).visitClassOrInterfaceOrEnumDeclaration(innerType); - unindent(); + indentedBlock(() -> new ClassOrInterfaceVisitor(parentNav) + .visitClassOrInterfaceOrEnumDeclaration(innerType)); } }); } @@ -1404,16 +1409,14 @@ private void getTypeDFS(Node node) { } private void addDefaultConstructor(TypeDeclaration typeDeclaration) { - indent(); - - addToken(INDENT, new Token(KEYWORD, "public"), SPACE); - final String name = typeDeclaration.getNameAsString(); - final String definitionId = makeId(typeDeclaration.getNameAsString()); - addToken(new Token(MEMBER_NAME, name, definitionId)); - addToken(new Token(PUNCTUATION, "(")); - addToken(new Token(PUNCTUATION, ")"), NEWLINE); - - unindent(); + indentedBlock(() -> { + addToken(INDENT, new Token(KEYWORD, "public"), SPACE); + final String name = typeDeclaration.getNameAsString(); + final String definitionId = makeId(typeDeclaration.getNameAsString()); + addToken(new Token(MEMBER_NAME, name, definitionId)); + addToken(new Token(PUNCTUATION, "(")); + addToken(new Token(PUNCTUATION, ")"), NEWLINE); + }); } } @@ -1518,22 +1521,33 @@ private void visitJavaDoc(JavadocComment javadoc) { addToken(new Token(DOCUMENTATION_RANGE_END)); } - private void indent() { - indent += 4; + private void indentedBlock(Runnable indentedSection) { + indentLevel++; + indentedSection.run(); + indentLevel--; } - private void unindent() { - indent = Math.max(indent - 4, 0); - } + // Keep a list of indent levels so we don't need to recreate a string each time indenting is done. + // Start with an initial set of levels. + private static final List INDENT_LEVELS = new ArrayList<>(Arrays.asList("", makeWhitespace(4), + makeWhitespace(8), makeWhitespace(12), makeWhitespace(16), makeWhitespace(20), makeWhitespace(24))); private Token makeWhitespace() { // Use a byte array with Arrays.fill with empty space (' ') character rather than StringBuilder as StringBuilder // will check that it has sufficient size every time a new character is appended. We know ahead of time the size // needed and can remove all those checks by removing usage of StringBuilder with this simpler pattern. - byte[] bytes = new byte[indent]; + return new Token(WHITESPACE, makeWhitespace(indentLevel)); + } + + private static String makeWhitespace(int indentLevel) { + if (indentLevel < INDENT_LEVELS.size()) { + return INDENT_LEVELS.get(indentLevel); + } + + byte[] bytes = new byte[indentLevel * 4]; Arrays.fill(bytes, (byte) ' '); - return new Token(WHITESPACE, new String(bytes, StandardCharsets.UTF_8)); + return new String(bytes, StandardCharsets.UTF_8); } private void addComment(String comment) { From f82aa7bbd9a6e8b88cdbbff29a686ca8a97aac6d Mon Sep 17 00:00:00 2001 From: Alan Zimmer <48699787+alzimmermsft@users.noreply.github.com> Date: Wed, 17 Apr 2024 12:53:07 -0400 Subject: [PATCH 2/5] Add indentation method and fix Javadoc indentation --- .../processor/analysers/JavaASTAnalyser.java | 27 ++++++++++--------- 1 file changed, 15 insertions(+), 12 deletions(-) diff --git a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java index dfc0ba1bfe0..ab34e506102 100644 --- a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java +++ b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java @@ -106,7 +106,13 @@ public class JavaASTAnalyser implements Analyser { private static final boolean SHOW_JAVADOC = true; + // Keep a list of indent levels so we don't need to recreate a string each time indenting is done. + // Start with an initial set of levels. + private static final List INDENT_LEVELS; + private static final Map ANNOTATION_RULE_MAP; + private static final JavaParserAdapter JAVA_8_PARSER; + private static final JavaParserAdapter JAVA_11_PARSER; static { /* For some annotations, we want to customise how they are displayed. Sometimes, we don't show them in any @@ -120,12 +126,7 @@ public class JavaASTAnalyser implements Analyser { // we always want @Metadata annotations to be fully expanded, but in a condensed form ANNOTATION_RULE_MAP.put("Metadata", new AnnotationRule().setShowProperties(true).setCondensed(true)); - } - - private static final JavaParserAdapter JAVA_8_PARSER; - private static final JavaParserAdapter JAVA_11_PARSER; - static { // Configure JavaParser to use type resolution JAVA_8_PARSER = JavaParserAdapter.of(new JavaParser(new ParserConfiguration() .setLanguageLevel(ParserConfiguration.LanguageLevel.JAVA_8) @@ -134,6 +135,9 @@ public class JavaASTAnalyser implements Analyser { JAVA_11_PARSER = JavaParserAdapter.of(new JavaParser(new ParserConfiguration() .setLanguageLevel(ParserConfiguration.LanguageLevel.JAVA_11) .setDetectOriginalLineSeparator(false))); + + INDENT_LEVELS = new ArrayList<>(Arrays.asList("", makeNewWhitespace(4), makeNewWhitespace(8), + makeNewWhitespace(12), makeNewWhitespace(16), makeNewWhitespace(20), makeNewWhitespace(24))); } // This is the model that we build up as the AST of all files are analysed. The APIListing is then output as @@ -1485,7 +1489,7 @@ private void visitJavaDoc(JavadocComment javadoc) { // The updated configuration from getDeclarationAsString removes the comment option and hence the toString // returns an empty string now. So, here we are using the toString overload that takes a PrintConfiguration // to get the old behavior. - splitNewLine(javadoc.asString()).forEach(line -> { + splitNewLine(javadoc.toString()).forEach(line -> { // we want to wrap our javadocs so that they are easier to read, so we wrap at 120 chars MiscUtils.wrap(line, 120).forEach(line2 -> { if (line2.contains("&")) { @@ -1527,11 +1531,6 @@ private void indentedBlock(Runnable indentedSection) { indentLevel--; } - // Keep a list of indent levels so we don't need to recreate a string each time indenting is done. - // Start with an initial set of levels. - private static final List INDENT_LEVELS = new ArrayList<>(Arrays.asList("", makeWhitespace(4), - makeWhitespace(8), makeWhitespace(12), makeWhitespace(16), makeWhitespace(20), makeWhitespace(24))); - private Token makeWhitespace() { // Use a byte array with Arrays.fill with empty space (' ') character rather than StringBuilder as StringBuilder // will check that it has sufficient size every time a new character is appended. We know ahead of time the size @@ -1544,7 +1543,11 @@ private static String makeWhitespace(int indentLevel) { return INDENT_LEVELS.get(indentLevel); } - byte[] bytes = new byte[indentLevel * 4]; + return makeNewWhitespace(indentLevel * 4); + } + + private static String makeNewWhitespace(int size) { + byte[] bytes = new byte[size]; Arrays.fill(bytes, (byte) ' '); return new String(bytes, StandardCharsets.UTF_8); From 30d612462823a2e54aa4613621f6c92abd8c1c05 Mon Sep 17 00:00:00 2001 From: Alan Zimmer <48699787+alzimmermsft@users.noreply.github.com> Date: Wed, 17 Apr 2024 13:19:50 -0400 Subject: [PATCH 3/5] Also don't make annotations clickable --- .../tools/apiview/processor/analysers/JavaASTAnalyser.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java index ab34e506102..2a20244bee5 100644 --- a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java +++ b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java @@ -1101,7 +1101,7 @@ private void getAnnotations(final NodeWithAnnotations nodeWithAnnotations, private List renderAnnotation(AnnotationRendererModel m) { final AnnotationExpr a = m.getAnnotation(); List tokens = new ArrayList<>(); - tokens.add(new Token(TYPE_NAME, "@" + a.getNameAsString(), makeId(a, m.getAnnotationParent()))); + tokens.add(new Token(TYPE_NAME, "@" + a.getNameAsString())); if (m.isShowProperties()) { if (a instanceof NormalAnnotationExpr) { tokens.add(new Token(PUNCTUATION, "(")); From 46d7e4a4f0c0a7e3775d7d70a0d15d8f4749a047 Mon Sep 17 00:00:00 2001 From: Alan Zimmer <48699787+alzimmermsft@users.noreply.github.com> Date: Wed, 17 Apr 2024 14:49:19 -0400 Subject: [PATCH 4/5] Remove whitespace cache --- .../processor/analysers/JavaASTAnalyser.java | 23 ++++--------------- 1 file changed, 4 insertions(+), 19 deletions(-) diff --git a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java index 2a20244bee5..24496627f6d 100644 --- a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java +++ b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java @@ -106,10 +106,6 @@ public class JavaASTAnalyser implements Analyser { private static final boolean SHOW_JAVADOC = true; - // Keep a list of indent levels so we don't need to recreate a string each time indenting is done. - // Start with an initial set of levels. - private static final List INDENT_LEVELS; - private static final Map ANNOTATION_RULE_MAP; private static final JavaParserAdapter JAVA_8_PARSER; private static final JavaParserAdapter JAVA_11_PARSER; @@ -135,9 +131,6 @@ public class JavaASTAnalyser implements Analyser { JAVA_11_PARSER = JavaParserAdapter.of(new JavaParser(new ParserConfiguration() .setLanguageLevel(ParserConfiguration.LanguageLevel.JAVA_11) .setDetectOriginalLineSeparator(false))); - - INDENT_LEVELS = new ArrayList<>(Arrays.asList("", makeNewWhitespace(4), makeNewWhitespace(8), - makeNewWhitespace(12), makeNewWhitespace(16), makeNewWhitespace(20), makeNewWhitespace(24))); } // This is the model that we build up as the AST of all files are analysed. The APIListing is then output as @@ -1532,22 +1525,14 @@ private void indentedBlock(Runnable indentedSection) { } private Token makeWhitespace() { - // Use a byte array with Arrays.fill with empty space (' ') character rather than StringBuilder as StringBuilder - // will check that it has sufficient size every time a new character is appended. We know ahead of time the size - // needed and can remove all those checks by removing usage of StringBuilder with this simpler pattern. return new Token(WHITESPACE, makeWhitespace(indentLevel)); } private static String makeWhitespace(int indentLevel) { - if (indentLevel < INDENT_LEVELS.size()) { - return INDENT_LEVELS.get(indentLevel); - } - - return makeNewWhitespace(indentLevel * 4); - } - - private static String makeNewWhitespace(int size) { - byte[] bytes = new byte[size]; + // Use a byte array with Arrays.fill with empty space (' ') character rather than StringBuilder as StringBuilder + // will check that it has sufficient size every time a new character is appended. We know ahead of time the size + // needed and can remove all those checks by removing usage of StringBuilder with this simpler pattern. + byte[] bytes = new byte[indentLevel * 4]; Arrays.fill(bytes, (byte) ' '); return new String(bytes, StandardCharsets.UTF_8); From ab24bc095648ba52b75d0f224b1c8ee758e8e7e9 Mon Sep 17 00:00:00 2001 From: Alan Zimmer <48699787+alzimmermsft@users.noreply.github.com> Date: Wed, 17 Apr 2024 14:50:16 -0400 Subject: [PATCH 5/5] Remove commented out code --- .../processor/analysers/JavaASTAnalyser.java | 13 ------------- 1 file changed, 13 deletions(-) diff --git a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java index 24496627f6d..130d60d88b4 100644 --- a/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java +++ b/src/java/apiview-java-processor/src/main/java/com/azure/tools/apiview/processor/analysers/JavaASTAnalyser.java @@ -376,19 +376,6 @@ private void tokeniseMavenPom(Pom mavenPom) { addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); addToken(new Token(SKIP_DIFF_END)); - - // allowed dependencies (in maven-enforcer) - // if (!mavenPom.getAllowedDependencies().isEmpty()) { - // addToken(INDENT, new Token(KEYWORD, "allowed-dependencies"), SPACE); - // addToken(new Token(PUNCTUATION, "{"), NEWLINE); - // indent(); - // mavenPom.getAllowedDependencies().stream().forEach(value -> { - // addToken(INDENT, new Token(TEXT, value, value), NEWLINE); - // }); - // unindent(); - // addToken(INDENT, new Token(PUNCTUATION, "}"), NEWLINE); - // } - // close maven });