1 | |
package org.apache.maven.plugin.checkstyle; |
2 | |
|
3 | |
import java.util.ArrayList; |
4 | |
import java.util.Iterator; |
5 | |
import java.util.List; |
6 | |
|
7 | |
import org.apache.maven.plugin.AbstractMojo; |
8 | |
import org.apache.maven.plugin.MojoExecutionException; |
9 | |
|
10 | |
|
11 | |
|
12 | |
|
13 | |
|
14 | |
|
15 | |
|
16 | |
|
17 | |
|
18 | 0 | public class HelpMojo |
19 | |
extends AbstractMojo |
20 | |
{ |
21 | |
|
22 | |
|
23 | |
|
24 | |
|
25 | |
|
26 | |
private boolean detail; |
27 | |
|
28 | |
|
29 | |
|
30 | |
|
31 | |
|
32 | |
|
33 | |
private java.lang.String goal; |
34 | |
|
35 | |
|
36 | |
|
37 | |
|
38 | |
|
39 | |
|
40 | |
private int lineLength; |
41 | |
|
42 | |
|
43 | |
|
44 | |
|
45 | |
|
46 | |
|
47 | |
private int indentSize; |
48 | |
|
49 | |
|
50 | |
|
51 | |
public void execute() |
52 | |
throws MojoExecutionException |
53 | |
{ |
54 | 0 | if ( lineLength <= 0 ) |
55 | |
{ |
56 | 0 | getLog().warn( "The parameter 'lineLength' should be positive, using '80' as default." ); |
57 | 0 | lineLength = 80; |
58 | |
} |
59 | 0 | if ( indentSize <= 0 ) |
60 | |
{ |
61 | 0 | getLog().warn( "The parameter 'indentSize' should be positive, using '2' as default." ); |
62 | 0 | indentSize = 2; |
63 | |
} |
64 | |
|
65 | 0 | StringBuffer sb = new StringBuffer(); |
66 | |
|
67 | 0 | append( sb, "org.apache.maven.plugins:maven-checkstyle-plugin:2.4", 0 ); |
68 | 0 | append( sb, "", 0 ); |
69 | |
|
70 | 0 | append( sb, "Maven Checkstyle Plugin", 0 ); |
71 | 0 | append( sb, "Generates a report on violations of code style and optionally fails the build if violations are detected.", 1 ); |
72 | 0 | append( sb, "", 0 ); |
73 | |
|
74 | 0 | if ( goal == null || goal.length() <= 0 ) |
75 | |
{ |
76 | 0 | append( sb, "This plugin has 3 goals:", 0 ); |
77 | 0 | append( sb, "", 0 ); |
78 | |
} |
79 | |
|
80 | 0 | if ( goal == null || goal.length() <= 0 || "check".equals( goal ) ) |
81 | |
{ |
82 | 0 | append( sb, "checkstyle:check", 0 ); |
83 | 0 | append( sb, "Perform a violation check against the last Checkstyle run to see if there are any violations. It reads the Checkstyle output file, counts the number of violations found and displays it on the console.", 1 ); |
84 | 0 | append( sb, "", 0 ); |
85 | 0 | if ( detail ) |
86 | |
{ |
87 | 0 | append( sb, "Available parameters:", 1 ); |
88 | 0 | append( sb, "", 0 ); |
89 | |
|
90 | 0 | append( sb, "failOnViolation (Default: true)", 2 ); |
91 | 0 | append( sb, "Do we fail the build on a violation?", 3 ); |
92 | 0 | append( sb, "", 0 ); |
93 | |
|
94 | 0 | append( sb, "logViolationsToConsole (Default: false)", 2 ); |
95 | 0 | append( sb, "Output the detected violations to the console.", 3 ); |
96 | 0 | append( sb, "", 0 ); |
97 | |
|
98 | 0 | append( sb, "maxAllowedViolations (Default: 0)", 2 ); |
99 | 0 | append( sb, "The maximum number of allowed violations. The execution fails only if the number of violations is above this limit.", 3 ); |
100 | 0 | append( sb, "", 0 ); |
101 | |
|
102 | 0 | append( sb, "outputFile (Default: ${project.build.directory}/checkstyle-result.xml)", 2 ); |
103 | 0 | append( sb, "Specifies the path and filename to save the Checkstyle output. The format of the output file is determined by the outputFileFormat parameter.", 3 ); |
104 | 0 | append( sb, "", 0 ); |
105 | |
|
106 | 0 | append( sb, "outputFileFormat (Default: xml)", 2 ); |
107 | 0 | append( sb, "Specifies the format of the output to be used when writing to the output file. Valid values are \'plain\' and \'xml\'.", 3 ); |
108 | 0 | append( sb, "", 0 ); |
109 | |
|
110 | 0 | append( sb, "skip (Default: false)", 2 ); |
111 | 0 | append( sb, "Skip entire check.", 3 ); |
112 | 0 | append( sb, "", 0 ); |
113 | |
|
114 | 0 | append( sb, "violationSeverity (Default: error)", 2 ); |
115 | 0 | append( sb, "The lowest severity level that is considered a violation. Valid values are \'error\', \'warning\' and \'info\'.", 3 ); |
116 | 0 | append( sb, "", 0 ); |
117 | |
} |
118 | |
} |
119 | |
|
120 | 0 | if ( goal == null || goal.length() <= 0 || "checkstyle".equals( goal ) ) |
121 | |
{ |
122 | 0 | append( sb, "checkstyle:checkstyle", 0 ); |
123 | 0 | append( sb, "Perform a Checkstyle analysis, and generate a report on violations.", 1 ); |
124 | 0 | append( sb, "", 0 ); |
125 | 0 | if ( detail ) |
126 | |
{ |
127 | 0 | append( sb, "Available parameters:", 1 ); |
128 | 0 | append( sb, "", 0 ); |
129 | |
|
130 | 0 | append( sb, "cacheFile (Default: ${project.build.directory}/checkstyle-cachefile)", 2 ); |
131 | 0 | append( sb, "Specifies the cache file used to speed up Checkstyle on successive runs.", 3 ); |
132 | 0 | append( sb, "", 0 ); |
133 | |
|
134 | 0 | append( sb, "configLocation (Default: config/sun_checks.xml)", 2 ); |
135 | 0 | append( sb, "Specifies the location of the XML configuration to use.\n\nPotential values are a filesystem path, a URL, or a classpath resource. This parameter expects that the contents of the location conform to the xml format (Checkstyle Checker module) configuration of rulesets.\n\nThis parameter is resolved as resource, URL, then file. If successfully resolved, the contents of the configuration is copied into the ${project.build.directory}/checkstyle-configuration.xml file before being passed to Checkstyle as a configuration.\n\nThere are 4 predefined rulesets.\n\n-\tconfig/sun_checks.xml: Sun Checks.\n-\tconfig/turbine_checks.xml: Turbine Checks.\n-\tconfig/avalon_checks.xml: Avalon Checks.\n-\tconfig/maven_checks.xml: Maven Source Checks.\n", 3 ); |
136 | 0 | append( sb, "", 0 ); |
137 | |
|
138 | 0 | append( sb, "consoleOutput (Default: false)", 2 ); |
139 | 0 | append( sb, "Output errors to console.", 3 ); |
140 | 0 | append( sb, "", 0 ); |
141 | |
|
142 | 0 | append( sb, "enableFilesSummary (Default: true)", 2 ); |
143 | 0 | append( sb, "Specifies if the Files summary should be enabled or not.", 3 ); |
144 | 0 | append( sb, "", 0 ); |
145 | |
|
146 | 0 | append( sb, "enableRSS (Default: true)", 2 ); |
147 | 0 | append( sb, "Specifies if the RSS should be enabled or not.", 3 ); |
148 | 0 | append( sb, "", 0 ); |
149 | |
|
150 | 0 | append( sb, "enableRulesSummary (Default: true)", 2 ); |
151 | 0 | append( sb, "Specifies if the Rules summary should be enabled or not.", 3 ); |
152 | 0 | append( sb, "", 0 ); |
153 | |
|
154 | 0 | append( sb, "enableSeveritySummary (Default: true)", 2 ); |
155 | 0 | append( sb, "Specifies if the Severity summary should be enabled or not.", 3 ); |
156 | 0 | append( sb, "", 0 ); |
157 | |
|
158 | 0 | append( sb, "encoding (Default: ${project.build.sourceEncoding})", 2 ); |
159 | 0 | append( sb, "The file encoding to use when reading the source files. If the property project.build.sourceEncoding is not set, the platform default encoding is used. Note: This parameter always overrides the property charset from Checkstyle\'s TreeWalker module.", 3 ); |
160 | 0 | append( sb, "", 0 ); |
161 | |
|
162 | 0 | append( sb, "excludes", 2 ); |
163 | 0 | append( sb, "Specifies the names filter of the source files to be excluded for Checkstyle.", 3 ); |
164 | 0 | append( sb, "", 0 ); |
165 | |
|
166 | 0 | append( sb, "failsOnError (Default: false)", 2 ); |
167 | 0 | append( sb, "Specifies if the build should fail upon a violation.", 3 ); |
168 | 0 | append( sb, "", 0 ); |
169 | |
|
170 | 0 | append( sb, "format (Default: sun)", 2 ); |
171 | 0 | append( sb, "Deprecated. Use configLocation instead.", 3 ); |
172 | 0 | append( sb, "", 0 ); |
173 | 0 | append( sb, "Specifies what predefined check set to use. Available sets are \'sun\' (for the Sun coding conventions), \'turbine\', and \'avalon\'.", 3 ); |
174 | 0 | append( sb, "", 0 ); |
175 | |
|
176 | 0 | append( sb, "headerFile", 2 ); |
177 | 0 | append( sb, "Deprecated. Use headerLocation instead.", 3 ); |
178 | 0 | append( sb, "", 0 ); |
179 | 0 | append( sb, "Specifies the location of the License file (a.k.a. the header file) that is used by Checkstyle to verify that source code has the correct license header.", 3 ); |
180 | 0 | append( sb, "", 0 ); |
181 | |
|
182 | 0 | append( sb, "headerLocation (Default: LICENSE.txt)", 2 ); |
183 | 0 | append( sb, "Specifies the location of the License file (a.k.a. the header file) that can be used by Checkstyle to verify that source code has the correct license header.\n\nYou need to use ${checkstyle.header.file} in your Checkstyle xml configuration to reference the name of this header file.\n\nFor instance:\n\n<module name=\'RegexpHeader\'> <property name=\'headerFile\' value=\'${checkstyle.header.file}\'/> </module>\n", 3 ); |
184 | 0 | append( sb, "", 0 ); |
185 | |
|
186 | 0 | append( sb, "includes (Default: **/*.java)", 2 ); |
187 | 0 | append( sb, "Specifies the names filter of the source files to be used for Checkstyle.", 3 ); |
188 | 0 | append( sb, "", 0 ); |
189 | |
|
190 | 0 | append( sb, "includeTestSourceDirectory (Default: ${false})", 2 ); |
191 | 0 | append( sb, "Include or not the test source directory to be used for Checkstyle.", 3 ); |
192 | 0 | append( sb, "", 0 ); |
193 | |
|
194 | 0 | append( sb, "linkXRef (Default: true)", 2 ); |
195 | 0 | append( sb, "Link the violation line numbers to the source xref. Will link automatically if Maven JXR plugin is being used.", 3 ); |
196 | 0 | append( sb, "", 0 ); |
197 | |
|
198 | 0 | append( sb, "outputDirectory (Default: ${project.reporting.outputDirectory})", 2 ); |
199 | 0 | append( sb, "The output directory for the report. Note that this parameter is only evaluated if the goal is run directly from the command line. If the goal is run indirectly as part of a site generation, the output directory configured in Maven Site Plugin is used instead.", 3 ); |
200 | 0 | append( sb, "", 0 ); |
201 | |
|
202 | 0 | append( sb, "outputFile (Default: ${project.build.directory}/checkstyle-result.xml)", 2 ); |
203 | 0 | append( sb, "Specifies the path and filename to save the checkstyle output. The format of the output file is determined by the outputFileFormat parameter.", 3 ); |
204 | 0 | append( sb, "", 0 ); |
205 | |
|
206 | 0 | append( sb, "outputFileFormat (Default: xml)", 2 ); |
207 | 0 | append( sb, "Specifies the format of the output to be used when writing to the output file. Valid values are \'plain\' and \'xml\'.", 3 ); |
208 | 0 | append( sb, "", 0 ); |
209 | |
|
210 | 0 | append( sb, "packageNamesFile", 2 ); |
211 | 0 | append( sb, "Deprecated. Use packageNamesLocation instead.", 3 ); |
212 | 0 | append( sb, "", 0 ); |
213 | 0 | append( sb, "Specifies the location of the package names XML to be used to configure Checkstyle.", 3 ); |
214 | 0 | append( sb, "", 0 ); |
215 | |
|
216 | 0 | append( sb, "packageNamesLocation", 2 ); |
217 | 0 | append( sb, "Specifies the location of the package names XML to be used to configure the Checkstyle Packages.\n\nThis parameter is resolved as resource, URL, then file. If resolved to a resource, or a URL, the contents of the package names XML is copied into the ${project.build.directory}/checkstyle-packagenames.xml file before being passed to Checkstyle for loading.\n", 3 ); |
218 | 0 | append( sb, "", 0 ); |
219 | |
|
220 | 0 | append( sb, "propertiesFile", 2 ); |
221 | 0 | append( sb, "Deprecated. Use propertiesLocation instead.", 3 ); |
222 | 0 | append( sb, "", 0 ); |
223 | 0 | append( sb, "Specifies the location of the Checkstyle properties file that will be used to check the source.", 3 ); |
224 | 0 | append( sb, "", 0 ); |
225 | |
|
226 | 0 | append( sb, "propertiesLocation", 2 ); |
227 | 0 | append( sb, "Specifies the location of the properties file.\n\nThis parameter is resolved as URL, File then resource. If successfully resolved, the contents of the properties location is copied into the ${project.build.directory}/checkstyle-checker.properties file before being passed to Checkstyle for loading.\n\nThe contents of the propertiesLocation will be made available to Checkstyle for specifying values for parameters within the xml configuration (specified in the configLocation parameter).\n", 3 ); |
228 | 0 | append( sb, "", 0 ); |
229 | |
|
230 | 0 | append( sb, "propertiesURL", 2 ); |
231 | 0 | append( sb, "Deprecated. Use propertiesLocation instead.", 3 ); |
232 | 0 | append( sb, "", 0 ); |
233 | 0 | append( sb, "Specifies the URL of the Checkstyle properties that will be used to check the source.", 3 ); |
234 | 0 | append( sb, "", 0 ); |
235 | |
|
236 | 0 | append( sb, "propertyExpansion", 2 ); |
237 | 0 | append( sb, "Allows for specifying raw property expansion information.", 3 ); |
238 | 0 | append( sb, "", 0 ); |
239 | |
|
240 | 0 | append( sb, "skip (Default: false)", 2 ); |
241 | 0 | append( sb, "Skip entire check.", 3 ); |
242 | 0 | append( sb, "", 0 ); |
243 | |
|
244 | 0 | append( sb, "sourceDirectory (Default: ${project.build.sourceDirectory})", 2 ); |
245 | 0 | append( sb, "Specifies the location of the source directory to be used for Checkstyle.", 3 ); |
246 | 0 | append( sb, "", 0 ); |
247 | |
|
248 | 0 | append( sb, "suppressionsFile", 2 ); |
249 | 0 | append( sb, "Deprecated. Use suppressionsLocation instead.", 3 ); |
250 | 0 | append( sb, "", 0 ); |
251 | 0 | append( sb, "Specifies the location of the suppressions XML file to use. The plugin defines a Checkstyle property named checkstyle.suppressions.file with the value of this property. This allows using the Checkstyle property in your own custom checkstyle configuration file when specifying a suppressions file.", 3 ); |
252 | 0 | append( sb, "", 0 ); |
253 | |
|
254 | 0 | append( sb, "suppressionsFileExpression (Default: checkstyle.suppressions.file)", 2 ); |
255 | 0 | append( sb, "The key to be used in the properties for the suppressions file.", 3 ); |
256 | 0 | append( sb, "", 0 ); |
257 | |
|
258 | 0 | append( sb, "suppressionsLocation", 2 ); |
259 | 0 | append( sb, "Specifies the location of the suppressions XML file to use.\n\nThis parameter is resolved as resource, URL, then file. If successfully resolved, the contents of the suppressions XML is copied into the ${project.build.directory}/checkstyle-supressions.xml file before being passed to Checkstyle for loading.\n\nSee suppressionsFileExpression for the property that will be made available to your checkstyle configuration.\n", 3 ); |
260 | 0 | append( sb, "", 0 ); |
261 | |
|
262 | 0 | append( sb, "testSourceDirectory (Default: ${project.build.testSourceDirectory})", 2 ); |
263 | 0 | append( sb, "Specifies the location of the test source directory to be used for Checkstyle.", 3 ); |
264 | 0 | append( sb, "", 0 ); |
265 | |
|
266 | 0 | append( sb, "useFile", 2 ); |
267 | 0 | append( sb, "If null, the Checkstyle plugin will display violations on stdout. Otherwise, a text file will be created with the violations.", 3 ); |
268 | 0 | append( sb, "", 0 ); |
269 | |
|
270 | 0 | append( sb, "xrefLocation (Default: ${project.reporting.outputDirectory}/xref)", 2 ); |
271 | 0 | append( sb, "Location of the Xrefs to link to.", 3 ); |
272 | 0 | append( sb, "", 0 ); |
273 | |
} |
274 | |
} |
275 | |
|
276 | 0 | if ( goal == null || goal.length() <= 0 || "help".equals( goal ) ) |
277 | |
{ |
278 | 0 | append( sb, "checkstyle:help", 0 ); |
279 | 0 | append( sb, "Display help information on maven-checkstyle-plugin.\nCall\n\u00a0\u00a0mvn\u00a0checkstyle:help\u00a0-Ddetail=true\u00a0-Dgoal=<goal-name>\nto display parameter details.", 1 ); |
280 | 0 | append( sb, "", 0 ); |
281 | 0 | if ( detail ) |
282 | |
{ |
283 | 0 | append( sb, "Available parameters:", 1 ); |
284 | 0 | append( sb, "", 0 ); |
285 | |
|
286 | 0 | append( sb, "detail (Default: false)", 2 ); |
287 | 0 | append( sb, "If true, display all settable properties for each goal.", 3 ); |
288 | 0 | append( sb, "", 0 ); |
289 | |
|
290 | 0 | append( sb, "goal", 2 ); |
291 | 0 | append( sb, "The name of the goal for which to show help. If unspecified, all goals will be displayed.", 3 ); |
292 | 0 | append( sb, "", 0 ); |
293 | |
|
294 | 0 | append( sb, "indentSize (Default: 2)", 2 ); |
295 | 0 | append( sb, "The number of spaces per indentation level, should be positive.", 3 ); |
296 | 0 | append( sb, "", 0 ); |
297 | |
|
298 | 0 | append( sb, "lineLength (Default: 80)", 2 ); |
299 | 0 | append( sb, "The maximum length of a display line, should be positive.", 3 ); |
300 | 0 | append( sb, "", 0 ); |
301 | |
} |
302 | |
} |
303 | |
|
304 | 0 | if ( getLog().isInfoEnabled() ) |
305 | |
{ |
306 | 0 | getLog().info( sb.toString() ); |
307 | |
} |
308 | 0 | } |
309 | |
|
310 | |
|
311 | |
|
312 | |
|
313 | |
|
314 | |
|
315 | |
|
316 | |
|
317 | |
|
318 | |
|
319 | |
private static String repeat( String str, int repeat ) |
320 | |
{ |
321 | 0 | StringBuffer buffer = new StringBuffer( repeat * str.length() ); |
322 | |
|
323 | 0 | for ( int i = 0; i < repeat; i++ ) |
324 | |
{ |
325 | 0 | buffer.append( str ); |
326 | |
} |
327 | |
|
328 | 0 | return buffer.toString(); |
329 | |
} |
330 | |
|
331 | |
|
332 | |
|
333 | |
|
334 | |
|
335 | |
|
336 | |
|
337 | |
|
338 | |
|
339 | |
private void append( StringBuffer sb, String description, int indent ) |
340 | |
{ |
341 | 0 | for ( Iterator it = toLines( description, indent, indentSize, lineLength ).iterator(); it.hasNext(); ) |
342 | |
{ |
343 | 0 | sb.append( it.next().toString() ).append( '\n' ); |
344 | |
} |
345 | 0 | } |
346 | |
|
347 | |
|
348 | |
|
349 | |
|
350 | |
|
351 | |
|
352 | |
|
353 | |
|
354 | |
|
355 | |
|
356 | |
|
357 | |
private static List toLines( String text, int indent, int indentSize, int lineLength ) |
358 | |
{ |
359 | 0 | List lines = new ArrayList(); |
360 | |
|
361 | 0 | String ind = repeat( "\t", indent ); |
362 | 0 | String[] plainLines = text.split( "(\r\n)|(\r)|(\n)" ); |
363 | 0 | for ( int i = 0; i < plainLines.length; i++ ) |
364 | |
{ |
365 | 0 | toLines( lines, ind + plainLines[i], indentSize, lineLength ); |
366 | |
} |
367 | |
|
368 | 0 | return lines; |
369 | |
} |
370 | |
|
371 | |
|
372 | |
|
373 | |
|
374 | |
|
375 | |
|
376 | |
|
377 | |
|
378 | |
|
379 | |
private static void toLines( List lines, String line, int indentSize, int lineLength ) |
380 | |
{ |
381 | 0 | int lineIndent = getIndentLevel( line ); |
382 | 0 | StringBuffer buf = new StringBuffer( 256 ); |
383 | 0 | String[] tokens = line.split( " +" ); |
384 | 0 | for ( int i = 0; i < tokens.length; i++ ) |
385 | |
{ |
386 | 0 | String token = tokens[i]; |
387 | 0 | if ( i > 0 ) |
388 | |
{ |
389 | 0 | if ( buf.length() + token.length() >= lineLength ) |
390 | |
{ |
391 | 0 | lines.add( buf.toString() ); |
392 | 0 | buf.setLength( 0 ); |
393 | 0 | buf.append( repeat( " ", lineIndent * indentSize ) ); |
394 | |
} |
395 | |
else |
396 | |
{ |
397 | 0 | buf.append( ' ' ); |
398 | |
} |
399 | |
} |
400 | 0 | for ( int j = 0; j < token.length(); j++ ) |
401 | |
{ |
402 | 0 | char c = token.charAt( j ); |
403 | 0 | if ( c == '\t' ) |
404 | |
{ |
405 | 0 | buf.append( repeat( " ", indentSize - buf.length() % indentSize ) ); |
406 | |
} |
407 | 0 | else if ( c == '\u00A0' ) |
408 | |
{ |
409 | 0 | buf.append( ' ' ); |
410 | |
} |
411 | |
else |
412 | |
{ |
413 | 0 | buf.append( c ); |
414 | |
} |
415 | |
} |
416 | |
} |
417 | 0 | lines.add( buf.toString() ); |
418 | 0 | } |
419 | |
|
420 | |
|
421 | |
|
422 | |
|
423 | |
|
424 | |
|
425 | |
|
426 | |
private static int getIndentLevel( String line ) |
427 | |
{ |
428 | 0 | int level = 0; |
429 | 0 | for ( int i = 0; i < line.length() && line.charAt( i ) == '\t'; i++ ) |
430 | |
{ |
431 | 0 | level++; |
432 | |
} |
433 | 0 | for ( int i = level + 1; i <= level + 4 && i < line.length(); i++ ) |
434 | |
{ |
435 | 0 | if ( line.charAt( i ) == '\t' ) |
436 | |
{ |
437 | 0 | level++; |
438 | 0 | break; |
439 | |
} |
440 | |
} |
441 | 0 | return level; |
442 | |
} |
443 | |
} |