1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.tradefed.log; 18 19 import com.android.ddmlib.Log; 20 import com.android.ddmlib.Log.LogLevel; 21 import com.android.tradefed.config.GlobalConfiguration; 22 import com.android.tradefed.config.IGlobalConfiguration; 23 24 import java.io.PrintWriter; 25 import java.io.StringWriter; 26 import java.text.SimpleDateFormat; 27 import java.util.Date; 28 29 /** 30 * A logging utility class. Useful for code that needs to override static methods from {@link Log} 31 */ 32 public class LogUtil { 33 34 /** 35 * Make uninstantiable 36 */ LogUtil()37 private LogUtil() {} 38 39 /** 40 * Sent when a log message needs to be printed. This implementation prints the message to 41 * stdout in all cases. 42 * 43 * @param logLevel The {@link LogLevel} enum representing the priority of the message. 44 * @param tag The tag associated with the message. 45 * @param message The message to display. 46 */ printLog(LogLevel logLevel, String tag, String message)47 public static void printLog(LogLevel logLevel, String tag, String message) { 48 System.out.print(LogUtil.getLogFormatString(logLevel, tag, message)); 49 } 50 51 /** 52 * Creates a format string that is similar to the "threadtime" log format on the device. This 53 * is specifically useful because it includes the day and month (to differentiate times for 54 * long-running TF instances), and also uses 24-hour time to disambiguate morning from evening. 55 * <p/> 56 * @see Log#getLogFormatString(LogLevel, String, String) 57 */ getLogFormatString(LogLevel logLevel, String tag, String message)58 public static String getLogFormatString(LogLevel logLevel, String tag, String message) { 59 SimpleDateFormat formatter = new SimpleDateFormat("MM-dd HH:mm:ss"); 60 return String.format("%s %c/%s: %s\n", formatter.format(new Date()), 61 logLevel.getPriorityLetter(), tag, message); 62 } 63 64 /** 65 * A shim class for {@link Log} that automatically uses the simple classname of the caller as 66 * the log tag 67 */ 68 public static class CLog { 69 70 protected static final String CLASS_NAME = CLog.class.getName(); 71 private static IGlobalConfiguration sGlobalConfig = null; 72 73 /** 74 * The shim version of {@link Log#v(String, String)}. 75 * 76 * @param message The {@code String} to log 77 */ v(String message)78 public static void v(String message) { 79 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 80 Log.v(getClassName(2), message); 81 } 82 83 /** 84 * The shim version of {@link Log#v(String, String)}. Also calls String.format for 85 * convenience. 86 * 87 * @param format A format string for the message to log 88 * @param args The format string arguments 89 */ v(String format, Object... args)90 public static void v(String format, Object... args) { 91 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 92 Log.v(getClassName(2), String.format(format, args)); 93 } 94 95 /** 96 * The shim version of {@link Log#d(String, String)}. 97 * 98 * @param message The {@code String} to log 99 */ d(String message)100 public static void d(String message) { 101 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 102 Log.d(getClassName(2), message); 103 } 104 105 /** 106 * The shim version of {@link Log#d(String, String)}. Also calls String.format for 107 * convenience. 108 * 109 * @param format A format string for the message to log 110 * @param args The format string arguments 111 */ d(String format, Object... args)112 public static void d(String format, Object... args) { 113 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 114 Log.d(getClassName(2), String.format(format, args)); 115 } 116 117 /** 118 * The shim version of {@link Log#i(String, String)}. 119 * 120 * @param message The {@code String} to log 121 */ i(String message)122 public static void i(String message) { 123 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 124 Log.i(getClassName(2), message); 125 } 126 127 /** 128 * The shim version of {@link Log#i(String, String)}. Also calls String.format for 129 * convenience. 130 * 131 * @param format A format string for the message to log 132 * @param args The format string arguments 133 */ i(String format, Object... args)134 public static void i(String format, Object... args) { 135 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 136 Log.i(getClassName(2), String.format(format, args)); 137 } 138 139 /** 140 * The shim version of {@link Log#w(String, String)}. 141 * 142 * @param message The {@code String} to log 143 */ w(String message)144 public static void w(String message) { 145 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 146 Log.w(getClassName(2), message); 147 } 148 149 /** 150 * A variation of {@link Log#w(String, String)}, where the stack trace of provided 151 * {@link Throwable} is formatted and logged. 152 * 153 * @param t The {@link Throwable} to log 154 */ w(Throwable t)155 public static void w(Throwable t) { 156 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 157 Log.w(getClassName(2), getStackTraceString(t)); 158 } 159 160 /** 161 * The shim version of {@link Log#w(String, String)}. Also calls String.format for 162 * convenience. 163 * 164 * @param format A format string for the message to log 165 * @param args The format string arguments 166 */ w(String format, Object... args)167 public static void w(String format, Object... args) { 168 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 169 Log.w(getClassName(2), String.format(format, args)); 170 } 171 172 /** 173 * The shim version of {@link Log#e(String, String)}. 174 * 175 * @param message The {@code String} to log 176 */ e(String message)177 public static void e(String message) { 178 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 179 Log.e(getClassName(2), message); 180 } 181 182 /** 183 * The shim version of {@link Log#e(String, String)}. Also calls String.format for 184 * convenience. 185 * 186 * @param format A format string for the message to log 187 * @param args The format string arguments 188 */ e(String format, Object... args)189 public static void e(String format, Object... args) { 190 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 191 Log.e(getClassName(2), String.format(format, args)); 192 } 193 194 /** 195 * The shim version of {@link Log#e(String, Throwable)}. 196 * 197 * @param t the {@link Throwable} to output. 198 */ e(Throwable t)199 public static void e(Throwable t) { 200 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 201 Log.e(getClassName(2), t); 202 } 203 204 /** 205 * The shim version of {@link Log#logAndDisplay(LogLevel, String, String)}. 206 * 207 * @param logLevel the {@link LogLevel} 208 * @param message The {@code String} to log 209 */ logAndDisplay(LogLevel logLevel, String message)210 public static void logAndDisplay(LogLevel logLevel, String message) { 211 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 212 Log.logAndDisplay(logLevel, getClassName(2), message); 213 } 214 215 /** 216 * The shim version of {@link Log#logAndDisplay(LogLevel, String, String)}. 217 * 218 * @param logLevel the {@link LogLevel} 219 * @param format A format string for the message to log 220 * @param args The format string arguments 221 */ logAndDisplay(LogLevel logLevel, String format, Object... args)222 public static void logAndDisplay(LogLevel logLevel, String format, Object... args) { 223 // frame 2: skip frames 0 (#getClassName) and 1 (this method) 224 Log.logAndDisplay(logLevel, getClassName(2), String.format(format, args)); 225 } 226 227 /** 228 * What a Terrible Failure: Report a condition that should never happen. 229 * The error will always be logged at level ASSERT with the call stack. 230 * 231 * @param message The message you would like logged. 232 */ wtf(String message)233 public static void wtf(String message) { 234 wtf(message, (Throwable) null); 235 } 236 237 /** 238 * What a Terrible Failure: Report a condition that should never happen. 239 * The error will always be logged at level ASSERT with the call stack. 240 * 241 * @param t (Optional) An exception to log. If null, only message will be logged. 242 */ wtf(Throwable t)243 public static void wtf(Throwable t) { 244 wtf(t.getMessage(), t); 245 } 246 247 /** 248 * What a Terrible Failure: Report a condition that should never happen. 249 * The error will always be logged at level ASSERT with the call stack. 250 * Also calls String.format for convenience. 251 * 252 * @param format A format string for the message to log 253 * @param args The format string arguments 254 */ wtf(String format, Object... args)255 public static void wtf(String format, Object... args) { 256 wtf(String.format(format, args), (Throwable) null); 257 } 258 259 /** 260 * What a Terrible Failure: Report a condition that should never happen. 261 * The error will always be logged at level ASSERT with the call stack. 262 * 263 * @param message The message you would like logged. 264 * @param t (Optional) An exception to log. If null, only message will be logged. 265 */ wtf(String message, Throwable t)266 public static void wtf(String message, Throwable t) { 267 ITerribleFailureHandler wtfHandler = getGlobalConfigInstance().getWtfHandler(); 268 269 /* since wtf(String, Throwable) can be called directly or through an overloaded 270 * method, ie wtf(String), the stack trace frame of the external class name that 271 * called CLog can vary, so we use findCallerClassName to find it */ 272 String tag = findCallerClassName(); 273 String logMessage = "WTF - " + message; 274 String stackTrace = getStackTraceString(t); 275 if (stackTrace.length() > 0) { 276 logMessage += "\n" + stackTrace; 277 } 278 279 Log.logAndDisplay(LogLevel.ASSERT, tag, logMessage); 280 if (wtfHandler != null) { 281 wtfHandler.onTerribleFailure(message, t); 282 } 283 } 284 285 /** 286 * Sets the GlobalConfiguration instance for CLog to use - exposed for unit testing 287 * 288 * @param globalConfig the GlobalConfiguration object for CLog to use 289 */ 290 // @VisibleForTesting setGlobalConfigInstance(IGlobalConfiguration globalConfig)291 public static void setGlobalConfigInstance(IGlobalConfiguration globalConfig) { 292 sGlobalConfig = globalConfig; 293 } 294 295 /** 296 * Gets the GlobalConfiguration instance, useful for unit testing 297 * 298 * @return the GlobalConfiguration singleton instance 299 */ getGlobalConfigInstance()300 private static IGlobalConfiguration getGlobalConfigInstance() { 301 if (sGlobalConfig == null) { 302 sGlobalConfig = GlobalConfiguration.getInstance(); 303 } 304 return sGlobalConfig; 305 } 306 307 /** 308 * A helper method that parses the stack trace string out of the 309 * throwable. 310 * 311 * @param t contains the stack trace information 312 * @return A {@link String} containing the stack trace of the throwable. 313 */ getStackTraceString(Throwable t)314 private static String getStackTraceString(Throwable t) { 315 if (t == null) { 316 return ""; 317 } 318 319 StringWriter sw = new StringWriter(); 320 PrintWriter pw = new PrintWriter(sw); 321 t.printStackTrace(pw); 322 pw.flush(); 323 return sw.toString(); 324 } 325 326 /** 327 * Return the simple classname from the {@code frame}th stack frame in the call path. 328 * Note: this method does <emph>not</emph> check array bounds for the stack trace length. 329 * 330 * @param frame The index of the stack trace frame to inspect for the class name 331 * @return The simple class name (or full-qualified if an error occurs getting a ref to the 332 * class) for the given element of the stack trace. 333 */ getClassName(int frame)334 public static String getClassName(int frame) { 335 StackTraceElement[] frames = (new Throwable()).getStackTrace(); 336 return parseClassName(frames[frame].getClassName()); 337 } 338 339 /** 340 * Finds the external class name that directly called a CLog method. 341 * 342 * @return The simple class name (or full-qualified if an error occurs getting a ref to 343 * the class) of the external class that called a CLog method, or "Unknown" if 344 * the stack trace is empty or only contains CLog class names. 345 */ findCallerClassName()346 public static String findCallerClassName() { 347 return findCallerClassName(null); 348 } 349 350 /** 351 * Finds the external class name that directly called a CLog method. 352 * 353 * @param t (Optional) the stack trace to search within, exposed for unit testing 354 * @return The simple class name (or full-qualified if an error occurs getting a ref to 355 * the class) of the external class that called a CLog method, or "Unknown" if 356 * the stack trace is empty or only contains CLog class names. 357 */ findCallerClassName(Throwable t)358 public static String findCallerClassName(Throwable t) { 359 String className = "Unknown"; 360 361 if (t == null) { 362 t = new Throwable(); 363 } 364 StackTraceElement[] frames = t.getStackTrace(); 365 if (frames.length == 0) { 366 return className; 367 } 368 369 // starting with the first frame's class name (this CLog class) 370 // keep iterating until a frame of a different class name is found 371 int f; 372 for (f = 0; f < frames.length; f++) { 373 className = frames[f].getClassName(); 374 if (!className.equals(CLASS_NAME)) { 375 break; 376 } 377 } 378 379 return parseClassName(className); 380 } 381 382 /** 383 * Parses the simple class name out of the full class name. If the formatting already 384 * looks like a simple class name, then just returns that. 385 * 386 * @param fullName the full class name to parse 387 * @return The simple class name 388 */ 389 // @VisibleForTesting parseClassName(String fullName)390 static String parseClassName(String fullName) { 391 int lastdot = fullName.lastIndexOf('.'); 392 String simpleName = fullName; 393 if (lastdot != -1) { 394 simpleName = fullName.substring(lastdot + 1); 395 } 396 // handle inner class names 397 int lastdollar = simpleName.lastIndexOf('$'); 398 if (lastdollar != -1) { 399 simpleName = simpleName.substring(0, lastdollar); 400 } 401 return simpleName; 402 } 403 } 404 } 405