1 /*
2 * Licensed to the Apache Software Foundation (ASF) under one or more
3 * contributor license agreements. See the NOTICE file distributed with
4 * this work for additional information regarding copyright ownership.
5 * The ASF licenses this file to You under the Apache license, Version 2.0
6 * (the "License"); you may not use this file except in compliance with
7 * the License. You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the license for the specific language governing permissions and
15 * limitations under the license.
16 */
17 package org.apache.logging.log4j;
18
19 import java.io.Serializable;
20 import java.util.Collection;
21 import java.util.Locale;
22 import java.util.concurrent.ConcurrentHashMap;
23 import java.util.concurrent.ConcurrentMap;
24
25 import org.apache.logging.log4j.spi.StandardLevel;
26
27 /**
28 * Levels used for identifying the severity of an event. Levels are organized from most specific to least:
29 * <ul>
30 * <li>{@link #OFF} (most specific, no logging)</li>
31 * <li>{@link #FATAL} (most specific, little data)</li>
32 * <li>{@link #ERROR}</li>
33 * <li>{@link #WARN}</li>
34 * <li>{@link #INFO}</li>
35 * <li>{@link #DEBUG}</li>
36 * <li>{@link #TRACE} (least specific, a lot of data)</li>
37 * <li>{@link #ALL} (least specific, all data)</li>
38 * </ul>
39 *
40 * Typically, configuring a level in a filter or on a logger will cause logging events of that level and those
41 * that are more specific to pass through the filter.
42 * A special level, {@link #ALL}, is guaranteed to capture all levels when used in logging configurations.
43 */
44 public final class Level implements Comparable<Level>, Serializable {
45
46 private static final long serialVersionUID = 1581082L;
47 private static final ConcurrentMap<String, Level> levels = new ConcurrentHashMap<String, Level>();
48
49 /**
50 * No events will be logged.
51 */
52 public static final Level OFF;
53
54 /**
55 * A severe error that will prevent the application from continuing.
56 */
57 public static final Level FATAL;
58
59 /**
60 * An error in the application, possibly recoverable.
61 */
62 public static final Level ERROR;
63
64 /**
65 * An event that might possible lead to an error.
66 */
67 public static final Level WARN;
68
69 /**
70 * An event for informational purposes.
71 */
72 public static final Level INFO;
73
74 /**
75 * A general debugging event.
76 */
77 public static final Level DEBUG;
78
79 /**
80 * A fine-grained debug message, typically capturing the flow through the application.
81 */
82 public static final Level TRACE;
83
84 /**
85 * All events should be logged.
86 */
87 public static final Level ALL;
88
89 static {
90 OFF = new Level("OFF", StandardLevel.OFF.intLevel());
91 FATAL = new Level("FATAL", StandardLevel.FATAL.intLevel());
92 ERROR = new Level("ERROR", StandardLevel.ERROR.intLevel());
93 WARN = new Level("WARN", StandardLevel.WARN.intLevel());
94 INFO = new Level("INFO", StandardLevel.INFO.intLevel());
95 DEBUG = new Level("DEBUG", StandardLevel.DEBUG.intLevel());
96 TRACE = new Level("TRACE", StandardLevel.TRACE.intLevel());
97 ALL = new Level("ALL", StandardLevel.ALL.intLevel());
98 }
99
100 /**
101 * @since 2.1
102 */
103 public static final String CATEGORY = "Level";
104
105 private final String name;
106 private final int intLevel;
107 private final StandardLevel standardLevel;
108
109 private Level(final String name, final int intLevel) {
110 if (name == null || name.isEmpty()) {
111 throw new IllegalArgumentException("Illegal null Level constant");
112 }
113 if (intLevel < 0) {
114 throw new IllegalArgumentException("Illegal Level int less than zero.");
115 }
116 this.name = name;
117 this.intLevel = intLevel;
118 this.standardLevel = StandardLevel.getStandardLevel(intLevel);
119 if (levels.putIfAbsent(name, this) != null) {
120 throw new IllegalStateException("Level " + name + " has already been defined.");
121 }
122 }
123
124 /**
125 * Gets the integral value of this Level.
126 *
127 * @return the value of this Level.
128 */
129 public int intLevel() {
130 return this.intLevel;
131 }
132
133 /**
134 * Gets the standard Level values as an enum.
135 *
136 * @return an enum of the standard Levels.
137 */
138 public StandardLevel getStandardLevel() {
139 return standardLevel;
140 }
141
142 /**
143 * Compares this level against the level passed as an argument and returns true if this level is the same or is less
144 * specific.T
145 *
146 * @param level
147 * The level to test.
148 * @return True if this level Level is less specific or the same as the given Level.
149 */
150 public boolean isLessSpecificThan(final Level level) {
151 return this.intLevel >= level.intLevel;
152 }
153
154 /**
155 * Compares this level against the level passed as an argument and returns true if this level is the same or is more
156 * specific.
157 *
158 * @param level
159 * The level to test.
160 * @return True if this level Level is more specific or the same as the given Level.
161 */
162 public boolean isMoreSpecificThan(final Level level) {
163 return this.intLevel <= level.intLevel;
164 }
165
166 @Override
167 @SuppressWarnings("CloneDoesntCallSuperClone")
168 public Level clone() throws CloneNotSupportedException {
169 throw new CloneNotSupportedException();
170 }
171
172 @Override
173 public int compareTo(final Level other) {
174 return intLevel < other.intLevel ? -1 : (intLevel > other.intLevel ? 1 : 0);
175 }
176
177 @Override
178 public boolean equals(final Object other) {
179 return other instanceof Level && other == this;
180 }
181
182 public Class<Level> getDeclaringClass() {
183 return Level.class;
184 }
185
186 @Override
187 public int hashCode() {
188 return this.name.hashCode();
189 }
190
191 /**
192 * Gets the symbolic name of this Level. Equivalent to calling {@link #toString()}.
193 *
194 * @return the name of this Level.
195 */
196 public String name() {
197 return this.name;
198 }
199
200 @Override
201 public String toString() {
202 return this.name;
203 }
204
205 /**
206 * Retrieves an existing Level or creates on if it didn't previously exist.
207 * @param name The name of the level.
208 * @param intValue The integer value for the Level. If the level was previously created this value is ignored.
209 * @return The Level.
210 * @throws java.lang.IllegalArgumentException if the name is null or intValue is less than zero.
211 */
212 public static Level forName(final String name, final int intValue) {
213 final Level level = levels.get(name);
214 if (level != null) {
215 return level;
216 }
217 try {
218 return new Level(name, intValue);
219 } catch (final IllegalStateException ex) {
220 // The level was added by something else so just return that one.
221 return levels.get(name);
222 }
223 }
224
225 /**
226 * Return the Level associated with the name or null if the Level cannot be found.
227 * @param name The name of the Level.
228 * @return The Level or null.
229 */
230 public static Level getLevel(final String name) {
231 return levels.get(name);
232 }
233
234 /**
235 * Converts the string passed as argument to a level. If the
236 * conversion fails, then this method returns {@link #DEBUG}.
237 *
238 * @param sArg The name of the desired Level.
239 * @return The Level associated with the String.
240 */
241 public static Level toLevel(final String sArg) {
242 return toLevel(sArg, Level.DEBUG);
243 }
244
245 /**
246 * Converts the string passed as argument to a level. If the
247 * conversion fails, then this method returns the value of
248 * <code>defaultLevel</code>.
249 *
250 * @param name The name of the desired Level.
251 * @param defaultLevel The Level to use if the String is invalid.
252 * @return The Level associated with the String.
253 */
254 public static Level toLevel(final String name, final Level defaultLevel) {
255 if (name == null) {
256 return defaultLevel;
257 }
258 final Level level = levels.get(name.toUpperCase(Locale.ENGLISH));
259 return level == null ? defaultLevel : level;
260 }
261
262 /**
263 * Return an array of all the Levels that have been registered.
264 * @return An array of Levels.
265 */
266 public static Level[] values() {
267 final Collection<Level> values = Level.levels.values();
268 return values.toArray(new Level[values.size()]);
269 }
270
271 /**
272 * Return the Level associated with the name.
273 * @param name The name of the Level to return.
274 * @return The Level.
275 * @throws java.lang.NullPointerException if the Level name is {@code null}.
276 * @throws java.lang.IllegalArgumentException if the Level name is not registered.
277 */
278 public static Level valueOf(final String name) {
279 if (name == null) {
280 throw new NullPointerException("No level name given.");
281 }
282 final String levelName = name.toUpperCase(Locale.ENGLISH);
283 if (levels.containsKey(levelName)) {
284 return levels.get(levelName);
285 }
286 throw new IllegalArgumentException("Unknown level constant [" + levelName + "].");
287 }
288
289 /**
290 * Returns the enum constant of the specified enum type with the specified name. The name must match exactly an
291 * identifier used to declare an enum constant in this type. (Extraneous whitespace characters are not permitted.)
292 *
293 * @param enumType the {@code Class} object of the enum type from which to return a constant
294 * @param name the name of the constant to return
295 * @param <T> The enum type whose constant is to be returned
296 * @return the enum constant of the specified enum type with the specified name
297 * @throws java.lang.IllegalArgumentException if the specified enum type has no constant with the specified name,
298 * or the specified class object does not represent an enum type
299 * @throws java.lang.NullPointerException if {@code enumType} or {@code name} are {@code null}
300 * @see java.lang.Enum#valueOf(Class, String)
301 */
302 public static <T extends Enum<T>> T valueOf(final Class<T> enumType, final String name) {
303 return Enum.valueOf(enumType, name);
304 }
305
306 // for deserialization
307 protected Object readResolve() {
308 return Level.valueOf(this.name);
309 }
310 }
311