001//////////////////////////////////////////////////////////////////////////////// 002// checkstyle: Checks Java source code for adherence to a set of rules. 003// Copyright (C) 2001-2022 the original author or authors. 004// 005// This library is free software; you can redistribute it and/or 006// modify it under the terms of the GNU Lesser General Public 007// License as published by the Free Software Foundation; either 008// version 2.1 of the License, or (at your option) any later version. 009// 010// This library is distributed in the hope that it will be useful, 011// but WITHOUT ANY WARRANTY; without even the implied warranty of 012// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 013// Lesser General Public License for more details. 014// 015// You should have received a copy of the GNU Lesser General Public 016// License along with this library; if not, write to the Free Software 017// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 018//////////////////////////////////////////////////////////////////////////////// 019 020package com.puppycrawl.tools.checkstyle; 021 022import java.io.File; 023import java.io.IOException; 024import java.io.PrintWriter; 025import java.nio.charset.StandardCharsets; 026import java.util.function.Consumer; 027import java.util.regex.Matcher; 028import java.util.regex.Pattern; 029 030import com.puppycrawl.tools.checkstyle.api.CheckstyleException; 031import com.puppycrawl.tools.checkstyle.api.DetailAST; 032import com.puppycrawl.tools.checkstyle.api.DetailNode; 033import com.puppycrawl.tools.checkstyle.api.JavadocTokenTypes; 034import com.puppycrawl.tools.checkstyle.api.TokenTypes; 035import com.puppycrawl.tools.checkstyle.utils.JavadocUtil; 036import picocli.CommandLine; 037import picocli.CommandLine.Command; 038import picocli.CommandLine.Option; 039import picocli.CommandLine.ParameterException; 040import picocli.CommandLine.Parameters; 041import picocli.CommandLine.ParseResult; 042 043/** 044 * This class is used internally in the build process to write a property file 045 * with short descriptions (the first sentences) of TokenTypes constants. 046 * Request: 724871 047 * For IDE plugins (like the eclipse plugin) it would be useful to have 048 * a programmatic access to the first sentence of the TokenType constants, 049 * so they can use them in their configuration gui. 050 * 051 * @noinspection UseOfSystemOutOrSystemErr, unused, ClassIndependentOfModule 052 */ 053public final class JavadocPropertiesGenerator { 054 055 /** 056 * This regexp is used to extract the first sentence from the text. 057 * The end of the sentence is determined by the symbol "period", "exclamation mark" or 058 * "question mark", followed by a space or the end of the text. 059 */ 060 private static final Pattern END_OF_SENTENCE_PATTERN = Pattern.compile( 061 "(([^.?!]|[.?!](?!\\s|$))*+[.?!])(\\s|$)"); 062 063 /** Max width of the usage help message for this command. */ 064 private static final int USAGE_HELP_WIDTH = 100; 065 066 /** 067 * Don't create instance of this class, use the {@link #main(String[])} method instead. 068 */ 069 private JavadocPropertiesGenerator() { 070 } 071 072 /** 073 * TokenTypes.properties generator entry point. 074 * 075 * @param args the command line arguments 076 * @throws CheckstyleException if parser or lexer failed or if there is an IO problem 077 **/ 078 public static void main(String... args) throws CheckstyleException { 079 final CliOptions cliOptions = new CliOptions(); 080 final CommandLine cmd = new CommandLine(cliOptions).setUsageHelpWidth(USAGE_HELP_WIDTH); 081 try { 082 final ParseResult parseResult = cmd.parseArgs(args); 083 if (parseResult.isUsageHelpRequested()) { 084 cmd.usage(System.out); 085 } 086 else { 087 writePropertiesFile(cliOptions); 088 } 089 } 090 catch (ParameterException ex) { 091 System.err.println(ex.getMessage()); 092 ex.getCommandLine().usage(System.err); 093 } 094 } 095 096 /** 097 * Creates the .properties file from a .java file. 098 * 099 * @param options the user-specified options 100 * @throws CheckstyleException if a javadoc comment can not be parsed 101 */ 102 private static void writePropertiesFile(CliOptions options) throws CheckstyleException { 103 try (PrintWriter writer = new PrintWriter(options.outputFile, 104 StandardCharsets.UTF_8.name())) { 105 final DetailAST top = JavaParser.parseFile(options.inputFile, 106 JavaParser.Options.WITH_COMMENTS).getFirstChild(); 107 final DetailAST objBlock = getClassBody(top); 108 if (objBlock != null) { 109 iteratePublicStaticIntFields(objBlock, writer::println); 110 } 111 } 112 catch (IOException ex) { 113 throw new CheckstyleException("Failed to write javadoc properties of '" 114 + options.inputFile + "' to '" + options.outputFile + "'", ex); 115 } 116 } 117 118 /** 119 * Walks over the type members and push the first javadoc sentence of every 120 * {@code public} {@code static} {@code int} field to the consumer. 121 * 122 * @param objBlock the OBJBLOCK of a class to iterate over its members 123 * @param consumer first javadoc sentence consumer 124 * @throws CheckstyleException if failed to parse a javadoc comment 125 */ 126 private static void iteratePublicStaticIntFields(DetailAST objBlock, Consumer<String> consumer) 127 throws CheckstyleException { 128 for (DetailAST member = objBlock.getFirstChild(); member != null; 129 member = member.getNextSibling()) { 130 if (isPublicStaticFinalIntField(member)) { 131 final DetailAST modifiers = member.findFirstToken(TokenTypes.MODIFIERS); 132 final String firstJavadocSentence = getFirstJavadocSentence(modifiers); 133 if (firstJavadocSentence != null) { 134 consumer.accept(getName(member) + "=" + firstJavadocSentence.trim()); 135 } 136 } 137 } 138 } 139 140 /** 141 * Finds the class body of the first class in the DetailAST. 142 * 143 * @param top AST to find the class body 144 * @return OBJBLOCK token if found; {@code null} otherwise 145 */ 146 private static DetailAST getClassBody(DetailAST top) { 147 DetailAST ast = top; 148 while (ast != null && ast.getType() != TokenTypes.CLASS_DEF) { 149 ast = ast.getNextSibling(); 150 } 151 DetailAST objBlock = null; 152 if (ast != null) { 153 objBlock = ast.findFirstToken(TokenTypes.OBJBLOCK); 154 } 155 return objBlock; 156 } 157 158 /** 159 * Checks that the DetailAST is a {@code public} {@code static} {@code final} {@code int} field. 160 * 161 * @param ast to process 162 * @return {@code true} if matches; {@code false} otherwise 163 */ 164 private static boolean isPublicStaticFinalIntField(DetailAST ast) { 165 boolean result = ast.getType() == TokenTypes.VARIABLE_DEF; 166 if (result) { 167 final DetailAST type = ast.findFirstToken(TokenTypes.TYPE); 168 final DetailAST arrayDeclarator = type.getFirstChild().getNextSibling(); 169 result = arrayDeclarator == null 170 && type.getFirstChild().getType() == TokenTypes.LITERAL_INT; 171 if (result) { 172 final DetailAST modifiers = ast.findFirstToken(TokenTypes.MODIFIERS); 173 result = modifiers.findFirstToken(TokenTypes.LITERAL_PUBLIC) != null 174 && modifiers.findFirstToken(TokenTypes.LITERAL_STATIC) != null 175 && modifiers.findFirstToken(TokenTypes.FINAL) != null; 176 } 177 } 178 return result; 179 } 180 181 /** 182 * Extracts the name of an ast. 183 * 184 * @param ast to extract the name 185 * @return the text content of the inner {@code TokenTypes.IDENT} node 186 */ 187 private static String getName(DetailAST ast) { 188 return ast.findFirstToken(TokenTypes.IDENT).getText(); 189 } 190 191 /** 192 * Extracts the first sentence as HTML formatted text from the comment of an DetailAST. 193 * The end of the sentence is determined by the symbol "period", "exclamation mark" or 194 * "question mark", followed by a space or the end of the text. Inline tags @code and @literal 195 * are converted to HTML code. 196 * 197 * @param ast to extract the first sentence 198 * @return the first sentence of the inner {@code TokenTypes.BLOCK_COMMENT_BEGIN} node 199 * or {@code null} if the first sentence is absent or malformed (does not end with period) 200 * @throws CheckstyleException if a javadoc comment can not be parsed or an unsupported inline 201 * tag found 202 */ 203 private static String getFirstJavadocSentence(DetailAST ast) throws CheckstyleException { 204 String firstSentence = null; 205 for (DetailAST child = ast.getFirstChild(); child != null && firstSentence == null; 206 child = child.getNextSibling()) { 207 // If there is an annotation, the javadoc comment will be a child of it. 208 if (child.getType() == TokenTypes.ANNOTATION) { 209 firstSentence = getFirstJavadocSentence(child); 210 } 211 // Otherwise, the javadoc comment will be right here. 212 else if (child.getType() == TokenTypes.BLOCK_COMMENT_BEGIN 213 && JavadocUtil.isJavadocComment(child)) { 214 final DetailNode tree = DetailNodeTreeStringPrinter.parseJavadocAsDetailNode(child); 215 firstSentence = getFirstJavadocSentence(tree); 216 } 217 } 218 return firstSentence; 219 } 220 221 /** 222 * Extracts the first sentence as HTML formatted text from a DetailNode. 223 * The end of the sentence is determined by the symbol "period", "exclamation mark" or 224 * "question mark", followed by a space or the end of the text. Inline tags @code and @literal 225 * are converted to HTML code. 226 * 227 * @param tree to extract the first sentence 228 * @return the first sentence of the node or {@code null} if the first sentence is absent or 229 * malformed (does not end with any of the end-of-sentence markers) 230 * @throws CheckstyleException if an unsupported inline tag found 231 */ 232 private static String getFirstJavadocSentence(DetailNode tree) throws CheckstyleException { 233 String firstSentence = null; 234 final StringBuilder builder = new StringBuilder(128); 235 for (DetailNode node : tree.getChildren()) { 236 if (node.getType() == JavadocTokenTypes.TEXT) { 237 final Matcher matcher = END_OF_SENTENCE_PATTERN.matcher(node.getText()); 238 if (matcher.find()) { 239 // Commit the sentence if an end-of-sentence marker is found. 240 firstSentence = builder.append(matcher.group(1)).toString(); 241 break; 242 } 243 // Otherwise append the whole line and look for an end-of-sentence marker 244 // on the next line. 245 builder.append(node.getText()); 246 } 247 else if (node.getType() == JavadocTokenTypes.JAVADOC_INLINE_TAG) { 248 formatInlineCodeTag(builder, node); 249 } 250 else { 251 formatHtmlElement(builder, node); 252 } 253 } 254 return firstSentence; 255 } 256 257 /** 258 * Converts inline code tag into HTML form. 259 * 260 * @param builder to append 261 * @param inlineTag to format 262 * @throws CheckstyleException if the inline javadoc tag is not a literal nor a code tag 263 */ 264 private static void formatInlineCodeTag(StringBuilder builder, DetailNode inlineTag) 265 throws CheckstyleException { 266 boolean wrapWithCodeTag = false; 267 for (DetailNode node : inlineTag.getChildren()) { 268 switch (node.getType()) { 269 case JavadocTokenTypes.CODE_LITERAL: 270 wrapWithCodeTag = true; 271 break; 272 // The text to append. 273 case JavadocTokenTypes.TEXT: 274 if (wrapWithCodeTag) { 275 builder.append("<code>").append(node.getText()).append("</code>"); 276 } 277 else { 278 builder.append(node.getText()); 279 } 280 break; 281 // Empty content tags. 282 case JavadocTokenTypes.LITERAL_LITERAL: 283 case JavadocTokenTypes.JAVADOC_INLINE_TAG_START: 284 case JavadocTokenTypes.JAVADOC_INLINE_TAG_END: 285 case JavadocTokenTypes.WS: 286 break; 287 default: 288 throw new CheckstyleException("Unsupported inline tag " 289 + JavadocUtil.getTokenName(node.getType())); 290 } 291 } 292 } 293 294 /** 295 * Concatenates the HTML text from AST of a JavadocTokenTypes.HTML_ELEMENT. 296 * 297 * @param builder to append 298 * @param node to format 299 */ 300 private static void formatHtmlElement(StringBuilder builder, DetailNode node) { 301 switch (node.getType()) { 302 case JavadocTokenTypes.START: 303 case JavadocTokenTypes.HTML_TAG_NAME: 304 case JavadocTokenTypes.END: 305 case JavadocTokenTypes.TEXT: 306 case JavadocTokenTypes.SLASH: 307 builder.append(node.getText()); 308 break; 309 default: 310 for (DetailNode child : node.getChildren()) { 311 formatHtmlElement(builder, child); 312 } 313 break; 314 } 315 } 316 317 /** 318 * Helper class encapsulating the command line options and positional parameters. 319 */ 320 @Command(name = "java com.puppycrawl.tools.checkstyle.JavadocPropertiesGenerator", 321 mixinStandardHelpOptions = true) 322 private static class CliOptions { 323 324 /** 325 * The command line option to specify the output file. 326 */ 327 @Option(names = "--destfile", required = true, description = "The output file.") 328 private File outputFile; 329 330 /** 331 * The command line positional parameter to specify the input file. 332 */ 333 @Parameters(index = "0", description = "The input file.") 334 private File inputFile; 335 } 336}