aboutsummaryrefslogtreecommitdiff
path: root/src/xdocs/writinglisteners.xml.vm
blob: 963dc77451a92f517339b4c78b262dea0e7c4283 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
<?xml version="1.0" encoding="UTF-8"?>

<document xmlns="http://maven.apache.org/XDOC/2.0"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/XDOC/2.0 http://maven.apache.org/xsd/xdoc-2.0.xsd">

  <head>
    <title>Writing Listeners</title>
    <script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.3/jquery.min.js"/>
    <script type="text/javascript" src="js/anchors.js"/>
    <script type="text/javascript" src="js/google-analytics.js"/>
    <link rel="icon" href="images/favicon.png" type="image/x-icon" />
    <link rel="shortcut icon" href="images/favicon.ico" type="image/ico" />
  </head>

  <body>

    <section name="Overview">
      <p>
        A Checkstyle listener monitors the progress of a <code>Checker</code> during the audit of files. The <code>Checker</code> notifies its attached listeners of
        significant events such as the start of the audit of a file and
        the logging of a Check error, and the listeners respond
        appropriately.  Any number of listeners can be attached to a
        <code> Checker</code>. An audit always adds one of
        the distribution listeners, <a
        href="apidocs/com/puppycrawl/tools/checkstyle/DefaultLogger.html">DefaultLogger</a>
        or <a
        href="apidocs/com/puppycrawl/tools/checkstyle/XMLLogger.html">XMLLogger</a>,
        to report events. A <code>DefaultLogger</code>
        produces simple text output for the events it receives, and a
        <code>XMLLogger</code> produces an XML document for
        its events.
      </p>

      <p>
        Listeners <code>DefaultLogger</code> and <code> XMLLogger</code> are sufficient for most
        Checkstyle users, but you may find a need for a custom
        listener. For example, a user has requested verbose output of
        progress information during a Checkstyle run. Another user would
        like to filter error events. This document explains how to write
        listeners for such tasks and how to integrate them in a Checker
        module. It also describes two custom listeners that are inspired
        by ANT listeners: a listener that is a wrapper for the Jakarta
        Commons Logging API, and a listener that sends its results via
        email.
      </p>

      <p>
        A listener is an implementation of the <a
        href="apidocs/com/puppycrawl/tools/checkstyle/api/AuditListener.html">AuditListener</a>
        interface. During an audit, a <code>Checker</code>
        informs its attached <code>AuditListeners</code> of
        six kinds of events: audit started/ended, file started/ended,
        and logging of an error/exception.
      </p>

      <p>
        An audit passes an event to a listener as an <a
        href="apidocs/com/puppycrawl/tools/checkstyle/api/AuditEvent.html">AuditEvent</a>.
        A file-related <code>AuditEvent</code> contains the
        name of that file. An <code>AuditEvent</code> for
        error logging has a message, a severity level, a message source
        such as the name of a <code>Check</code>, and file
        line and column numbers that may be relevant to the error. The
        notification of an exception to a <code>AuditListener</code> includes an error <code>AuditEvent</code> and the details of the
        exception. Here is a UML diagram for classes <code>AuditListener</code> and <code>AuditEvent</code>.
      </p>

      <img src="images/AuditListener.gif" width="381" height="488"
             alt="AuditListener UML diagram"/>

    </section>

    <section name="Writing Listeners">
      <p>
        A custom listener is an implementation of the <a
        href="apidocs/com/puppycrawl/tools/checkstyle/api/AuditListener.html">AuditListener</a>
        interface. If the listener has properties that can be set from a
        configuration, the listener must extend <a
        href="apidocs/com/puppycrawl/tools/checkstyle/api/AutomaticBean.html">AutomaticBean</a>.
        An <code>AutomaticBean</code> uses JavaBean
        introspection to set JavaBean properties.
      </p>

      <p>
        The custom listener that we demonstrate here is a verbose
        listener that simply prints each event notification to an output
        stream, and reports the number of errors per audited file and
        the total number of errors. The default output stream is <code>System.out</code>. In order to enable the
        specification of output to a file through property <code>file</code>, the class extends <code>AutomaticBean</code> and defines method <code>setFile(String)</code>.
      </p>

      <source>
package com.mycompany.listeners;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.io.PrintWriter;

import com.puppycrawl.tools.checkstyle.api.AuditEvent;
import com.puppycrawl.tools.checkstyle.api.AuditListener;
import com.puppycrawl.tools.checkstyle.api.AutomaticBean;
import com.puppycrawl.tools.checkstyle.api.SeverityLevel;

public class VerboseListener
    extends AutomaticBean
    implements AuditListener
{
    private PrintWriter mWriter = new PrintWriter(System.out);
    private boolean mCloseOut = false;
    private int mTotalErrors;
    private int mErrors;

    public void setFile(String aFileName)
        throws FileNotFoundException
    {
        final OutputStream out = new FileOutputStream(aFileName);
        mWriter = new PrintWriter(out);
        mCloseOut = true;
    }

    public void auditStarted(AuditEvent aEvt)
    {
        mTotalErrors = 0;
        mWriter.println("Audit started.");
    }

    public void auditFinished(AuditEvent aEvt)
    {
        mWriter.println("Audit finished. Total errors: " + mTotalErrors);
        mWriter.flush();
        if (mCloseOut) {
            mWriter.close();
        }
    }

    public void fileStarted(AuditEvent aEvt)
    {
        mErrors = 0;
        mWriter.println(
            "Started checking file '" + aEvt.getFileName() + "'.");
    }

    public void fileFinished(AuditEvent aEvt)
    {
        mWriter.println("Finished checking file '" + aEvt.getFileName()
            + "'. Errors: " + mErrors);
    }

    public void addError(AuditEvent aEvt)
    {
        printEvent(aEvt);
        if (SeverityLevel.ERROR.equals(aEvt.getSeverityLevel())) {
            mErrors++;
            mTotalErrors++;
        }
    }

    public void addException(AuditEvent aEvt, Throwable aThrowable)
    {
        printEvent(aEvt);
        aThrowable.printStackTrace(System.out);
        mErrors++;
        mTotalErrors++;
    }

    private void printEvent(AuditEvent aEvt)
    {
        mWriter.println("Logging error -"
            + " file: '" + aEvt.getFileName() + "'"
            + " line: " + aEvt.getLine()
            + " column: " + aEvt.getColumn()
            + " severity: " + aEvt.getSeverityLevel()
            + " message: " + aEvt.getMessage()
            + " source: " + aEvt.getSourceName());
    }
}
      </source>

      <p>
        A listener that filters error events could perform the filtering
        in methods <code> addError</code> and <code>addException</code>. As further examples of
        listeners, <a
        href="#CommonsLoggingListener">CommonsLoggingListener</a>
        reports its events through the Commons Logging API, and <a
        href="#MailLogger">MailLogger</a> e-mails the audit report of a
        <code>DefaultLogger</code>.
      </p>
    </section>

    <section name="Using Listeners">
      <p>
        To incorporate a custom listener in the set of listeners for a
        <code>Checker</code>, include a module element for
        the listener in the <a
        href="config.html#AuditListeners">configuration file</a>. For
        example, to configure a <code>Checker</code> so
        that it uses custom listener <code>VerboseListener</code> to print audit messages to a
        file named &quot;audit.txt&quot;, include the following module
        in the configuration file:
      </p>

      <source>
&lt;module name=&quot;com.mycompany.listeners.VerboseListener&quot;&gt;
    &lt;property name=&quot;file&quot; value=&quot;audit.txt&quot;/&gt;
&lt;/module&gt;
      </source>

      <p>
        Here is a truncated example of audit output from a <code> VerboseListener</code>:
      </p>

      <source>
Audit started.
Started checking file 'CommonsLoggingListener.java'.
Finished checking file 'CommonsLoggingListener.java'. Errors: 0
Started checking file 'MailLogger.java'.
Finished checking file 'MailLogger.java'. Errors: 0
Started checking file 'VerboseListener.java'.
Logging error - file: 'VerboseListener.java' line: 23 ...
Finished checking file 'VerboseListener.java'. Errors: 1
Audit finished. Total errors: 1
      </source>
    </section>

    <section name="Examples">
      <p>
        This section describes two examples based on <a
        href="http://ant.apache.org/">ANT</a> listeners. The first
        listener, <code> CommonsLoggingListener</code>,
        hands off events to the <a
        href="http://commons.apache.org/proper/commons-logging/">Apache
        Commons Logging</a> facade and the second, <code>MailLogger</code>, sends a report of an audit via
        e-mail. The discussion of these examples and how to use them is
        derived from material in <a
        href="https://www.manning.com/books/java-development-with-ant">&quot;Java Development
        with Ant&quot;</a> by Eric Hatcher and Steve Loughran, an
        excellent ANT book.
      </p>

      <a name="CommonsLoggingListener"/>
      <h4>CommonsLoggingListener</h4>

      <p>
        <a
        href="http://commons.apache.org/proper/commons-logging/">Apache
        Commons Logging</a> provides a facade for logging tools
        <a
        href="http://logging.apache.org/log4j/2.x/index.html">log4j</a>,
        , J2SE 1.4, and others. Checkstyle listener <code> CommonsLoggingListener</code> responds to an
        AuditEvent by handing it off to the current Commons Logging Log.
      </p>

      <p>
        The source code for <code>CommonsLoggingListener</code> is in distribution
        directory <code>https://github.com/checkstyle/contribution/examples/listeners</code>.  Notice that
        each <code>AuditListener</code> method that
        receives an <code> AuditEvent</code> calls a method
        for the Commons Logging log level corresponding to the
        Checkstyle <code>SeverityLevel</code> of the <code> AuditEvent</code>.
      </p>

      <p>
        The easiest way to use <code>CommonsLoggingListener</code> is to include <code>checkstyle-${projectVersion}-all.jar</code>
        in the classpath because that jar file contains all the Commons
        Logging classes.  The default Log under J2SE 1.4 is wrapper
        class <a
        href="http://commons.apache.org/proper/commons-logging/apidocs/org/apache/commons/logging/impl/Jdk14Logger.html">Jdk14Logger</a>.
        Under earlier Java versions, the default Log is a simple wrapper
        class, <a
        href="http://commons.apache.org/proper/commons-logging/apidocs/org/apache/commons/logging/impl/SimpleLog.html">SimpleLog</a>.
        Both default logging tools can be used directly from Commons
        Logging; if you need to use other tools such as log4j, then you
        must include the appropriate jar file(s) in the classpath.
      </p>

      <p>
        Logging configuration details for Jakarta Commons Logging are in
        the <a
        href="http://commons.apache.org/proper/commons-logging/">documentation</a>.
        As a simple example, assume that <code>log4j.jar</code> is in the classpath and the
        following <code>log4j.properties</code> file is
        in the current directory:
      </p>

      <source>
# Set root logger level to INFO and its only appender to A1.
log4j.rootLogger=INFO, A1

# A1 is set to be a ConsoleAppender.
log4j.appender.A1=org.apache.log4j.ConsoleAppender

# A1 uses PatternLayout.
log4j.appender.A1.layout=org.apache.log4j.PatternLayout
log4j.appender.A1.layout.ConversionPattern=%-5p %c %x- %m%n
      </source>

      <p>
        Running a Checkstyle audit with a <code>CommonsLoggingListener</code> yields this
        (abbreviated) output:
      </p>

      <source>
INFO  com.puppycrawl...Checker  - Audit started.
INFO  com.puppycrawl...Checker  - File "CommonsLoggingListener.java" started.
INFO  com.puppycrawl...Checker  - File "CommonsLoggingListener.java" finished.
INFO  com.puppycrawl...Checker  - File "MailLogger.java" started.
INFO  com.puppycrawl...Checker  - File "MailLogger.java" finished.
INFO  com.puppycrawl...Checker  - File "VerboseListener.java" started.
ERROR com.puppycrawl...ParenPadCheck  - Line: 23 Column: 28 ...
INFO  com.puppycrawl...Checker  - File "VerboseListener.java" finished.
INFO  com.puppycrawl...Checker  - Audit finished.
      </source>

      <a name="MailLogger"/>
      <h4>MailLogger</h4>
      <p>
        <code>MailLogger</code> sends an audit report in an
        email message.  The listener uses a <code>DefaultLogger</code> to prepare the text of the
        message. The listener obtains other message parameters such as
        <code>to</code> and <code>subject</code> from environment properties that
        can be read from a properties file.
      </p>

      <p>
        The source code for <code>CommonsLoggingListener</code> is in distribution
        directory <code>https://github.com/checkstyle/contribution/examples/listeners</code>.  This
        implementation uses the <a
        href="http://www.oracle.com/technetwork/java/javamail/index.html">JavaMail API</a> as
        the mail system, and you must include appropriate jar files in
        the classpath.
      </p>

      <p>
        As an example of using <code>MailLogger</code>, set
        system property <code>-DMailLogger.properties.file=MailLogger.properties</code>,
        so that <code>MailLogger</code> reads message
        parameters from file <code>MailLogger.properties</code> of the current
        directory:
      </p>

      <source>
MailLogger.from=user@example.org
MailLogger.failure.to=user@example.org
MailLogger.success.to=user@example.org
MailLogger.mailhost=localhost
      </source>
    </section>

    <section name="Huh? I can&#39;t figure it out!">
      <p>
        That&#39;s probably our fault, and it means that we have to
        provide better documentation. Please do not hesitate to ask
        questions on the user mailing list, this will help us to improve
        this document.  Please ask your questions as precisely as
        possible. We will not be able to answer questions like &quot;I
        want to write a listener but I don&#39;t know how, can you help
        me?&quot;. Tell us what you are trying to do (the purpose of the
        listener), what you have understood so far, and what exactly you
        are getting stuck on.
      </p>
    </section>

    <section name="Contributing">
      <p>
        We need <em>your</em> help to keep improving Checkstyle.
        Whenever you write a listener that you think is generally
        useful, please consider <a
        href="contributing.html">contributing</a> it to the Checkstyle
        community and submit it for inclusion in the next release of
        Checkstyle.
      </p>
    </section>
  </body>
</document>