1 /* 2 * Copyright (C) 2016 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 android.telecom.Logging; 18 19 import android.annotation.NonNull; 20 import android.telecom.Log; 21 import android.text.TextUtils; 22 import android.util.Pair; 23 24 import com.android.internal.annotations.VisibleForTesting; 25 import com.android.internal.util.IndentingPrintWriter; 26 27 import java.time.Instant; 28 import java.time.ZoneId; 29 import java.time.ZonedDateTime; 30 import java.time.format.DateTimeFormatter; 31 import java.util.ArrayList; 32 import java.util.Collections; 33 import java.util.Comparator; 34 import java.util.HashMap; 35 import java.util.IllegalFormatException; 36 import java.util.LinkedList; 37 import java.util.List; 38 import java.util.Locale; 39 import java.util.Map; 40 import java.util.concurrent.LinkedBlockingQueue; 41 42 /** 43 * A utility class that provides the ability to define Events that a subsystem deems important, and 44 * then relate those events to other events so that information can be extracted. For example, a 45 * START and FINISH event can be defined and when a START and then FINISH occurs in a sequence, the 46 * time it took to complete that sequence can be saved to be retrieved later. 47 * @hide 48 */ 49 50 public class EventManager { 51 52 public static final String TAG = "Logging.Events"; 53 @VisibleForTesting 54 public static final int DEFAULT_EVENTS_TO_CACHE = 10; // Arbitrarily chosen. 55 public static final DateTimeFormatter DATE_TIME_FORMATTER = 56 DateTimeFormatter.ofPattern("HH:mm:ss.SSS"); 57 58 public interface Loggable { 59 /** 60 * @return a unique String ID that will allow the Event to be recognized later in the logs. 61 */ getId()62 String getId(); 63 64 /** 65 * @return Formatted information about the state that will be printed out later in the logs. 66 */ getDescription()67 String getDescription(); 68 } 69 70 private final Map<Loggable, EventRecord> mCallEventRecordMap = new HashMap<>(); 71 private LinkedBlockingQueue<EventRecord> mEventRecords = 72 new LinkedBlockingQueue<>(DEFAULT_EVENTS_TO_CACHE); 73 74 private List<EventListener> mEventListeners = new ArrayList<>(); 75 76 public interface EventListener { 77 /** 78 * Notifies the implementation of this method that a new event record has been added. 79 * @param eventRecord Reference to the recently added EventRecord 80 */ eventRecordAdded(EventRecord eventRecord)81 void eventRecordAdded(EventRecord eventRecord); 82 } 83 84 private SessionManager.ISessionIdQueryHandler mSessionIdHandler; 85 /** 86 * Maps from request events to a list of possible response events. Used to track 87 * end-to-end timing for critical user-facing operations in Telecom. 88 */ 89 private final Map<String, List<TimedEventPair>> requestResponsePairs = new HashMap<>(); 90 91 private static final Object mSync = new Object(); 92 93 /** 94 * Stores the various events. 95 * Also stores all request-response pairs amongst the events. 96 */ 97 public static class TimedEventPair { 98 private static final long DEFAULT_TIMEOUT = 3000L; 99 100 String mRequest; 101 String mResponse; 102 String mName; 103 long mTimeoutMillis = DEFAULT_TIMEOUT; 104 TimedEventPair(String request, String response, String name)105 public TimedEventPair(String request, String response, String name) { 106 this.mRequest = request; 107 this.mResponse = response; 108 this.mName = name; 109 } 110 TimedEventPair(String request, String response, String name, long timeoutMillis)111 public TimedEventPair(String request, String response, String name, long timeoutMillis) { 112 this.mRequest = request; 113 this.mResponse = response; 114 this.mName = name; 115 this.mTimeoutMillis = timeoutMillis; 116 } 117 } 118 addRequestResponsePair(TimedEventPair p)119 public void addRequestResponsePair(TimedEventPair p) { 120 if (requestResponsePairs.containsKey(p.mRequest)) { 121 requestResponsePairs.get(p.mRequest).add(p); 122 } else { 123 ArrayList<TimedEventPair> responses = new ArrayList<>(); 124 responses.add(p); 125 requestResponsePairs.put(p.mRequest, responses); 126 } 127 } 128 129 public static class Event { 130 public String eventId; 131 public String sessionId; 132 public long time; 133 public Object data; 134 // String storing the date for display. This will be computed at the time/timezone when 135 // the event is recorded. 136 public final String timestampString; 137 Event(String eventId, String sessionId, long time, Object data)138 public Event(String eventId, String sessionId, long time, Object data) { 139 this.eventId = eventId; 140 this.sessionId = sessionId; 141 this.time = time; 142 timestampString = 143 ZonedDateTime.ofInstant(Instant.ofEpochMilli(time), ZoneId.systemDefault()) 144 .format(DATE_TIME_FORMATTER); 145 this.data = data; 146 } 147 } 148 149 public class EventRecord { 150 public class EventTiming extends TimedEvent<String> { 151 public String name; 152 public long time; 153 EventTiming(String name, long time)154 public EventTiming(String name, long time) { 155 this.name = name; 156 this.time = time; 157 } 158 getKey()159 public String getKey() { 160 return name; 161 } 162 getTime()163 public long getTime() { 164 return time; 165 } 166 } 167 168 private class PendingResponse { 169 String requestEventId; 170 long requestEventTimeMillis; 171 long timeoutMillis; 172 String name; 173 PendingResponse(String requestEventId, long requestEventTimeMillis, long timeoutMillis, String name)174 public PendingResponse(String requestEventId, long requestEventTimeMillis, 175 long timeoutMillis, String name) { 176 this.requestEventId = requestEventId; 177 this.requestEventTimeMillis = requestEventTimeMillis; 178 this.timeoutMillis = timeoutMillis; 179 this.name = name; 180 } 181 } 182 183 private final List<Event> mEvents = Collections.synchronizedList(new ArrayList<>()); 184 private final Loggable mRecordEntry; 185 EventRecord(Loggable recordEntry)186 public EventRecord(Loggable recordEntry) { 187 mRecordEntry = recordEntry; 188 } 189 getRecordEntry()190 public Loggable getRecordEntry() { 191 return mRecordEntry; 192 } 193 addEvent(String event, String sessionId, Object data)194 public void addEvent(String event, String sessionId, Object data) { 195 mEvents.add(new Event(event, sessionId, System.currentTimeMillis(), data)); 196 Log.i("Event", "RecordEntry %s: %s, %s", mRecordEntry.getId(), event, data); 197 } 198 getEvents()199 public List<Event> getEvents() { 200 return new ArrayList<>(mEvents); 201 } 202 extractEventTimings()203 public List<EventTiming> extractEventTimings() { 204 if (mEvents == null) { 205 return Collections.emptyList(); 206 } 207 208 ArrayList<EventTiming> result = new ArrayList<>(); 209 Map<String, PendingResponse> pendingResponses = new HashMap<>(); 210 synchronized (mEvents) { 211 for (Event event : mEvents) { 212 if (requestResponsePairs.containsKey(event.eventId)) { 213 // This event expects a response, so add that expected response to the maps 214 // of pending events. 215 for (EventManager.TimedEventPair p : requestResponsePairs.get( 216 event.eventId)) { 217 pendingResponses.put(p.mResponse, new PendingResponse(event.eventId, 218 event.time, p.mTimeoutMillis, p.mName)); 219 } 220 } 221 222 PendingResponse pendingResponse = pendingResponses.remove(event.eventId); 223 if (pendingResponse != null) { 224 long elapsedTime = event.time - pendingResponse.requestEventTimeMillis; 225 if (elapsedTime < pendingResponse.timeoutMillis) { 226 result.add(new EventTiming(pendingResponse.name, elapsedTime)); 227 } 228 } 229 } 230 } 231 232 return result; 233 } 234 dump(IndentingPrintWriter pw)235 public void dump(IndentingPrintWriter pw) { 236 pw.print(mRecordEntry.getDescription()); 237 238 pw.increaseIndent(); 239 // Iterate over copy of events so that this doesn't hold the lock for too long. 240 for (Event event : getEvents()) { 241 pw.print(event.timestampString); 242 pw.print(" - "); 243 pw.print(event.eventId); 244 if (event.data != null) { 245 pw.print(" ("); 246 Object data = event.data; 247 248 if (data instanceof Loggable) { 249 // If the data is another Loggable, then change the data to the 250 // Entry's Event ID instead. 251 EventRecord record = mCallEventRecordMap.get(data); 252 if (record != null) { 253 data = "RecordEntry " + record.mRecordEntry.getId(); 254 } 255 } 256 257 pw.print(data); 258 pw.print(")"); 259 } 260 if (!TextUtils.isEmpty(event.sessionId)) { 261 pw.print(":"); 262 pw.print(event.sessionId); 263 } 264 pw.println(); 265 } 266 267 pw.println("Timings (average for this call, milliseconds):"); 268 pw.increaseIndent(); 269 Map<String, Double> avgEventTimings = EventTiming.averageTimings(extractEventTimings()); 270 List<String> eventNames = new ArrayList<>(avgEventTimings.keySet()); 271 Collections.sort(eventNames); 272 for (String eventName : eventNames) { 273 pw.printf("%s: %.2f\n", eventName, avgEventTimings.get(eventName)); 274 } 275 pw.decreaseIndent(); 276 pw.decreaseIndent(); 277 } 278 } 279 EventManager(@onNull SessionManager.ISessionIdQueryHandler l)280 public EventManager(@NonNull SessionManager.ISessionIdQueryHandler l) { 281 mSessionIdHandler = l; 282 } 283 event(Loggable recordEntry, String event, Object data)284 public void event(Loggable recordEntry, String event, Object data) { 285 String currentSessionID = mSessionIdHandler.getSessionId(); 286 287 if (recordEntry == null) { 288 Log.i(TAG, "Non-call EVENT: %s, %s", event, data); 289 return; 290 } 291 synchronized (mEventRecords) { 292 if (!mCallEventRecordMap.containsKey(recordEntry)) { 293 EventRecord newRecord = new EventRecord(recordEntry); 294 addEventRecord(newRecord); 295 } 296 297 EventRecord record = mCallEventRecordMap.get(recordEntry); 298 record.addEvent(event, currentSessionID, data); 299 } 300 } 301 event(Loggable recordEntry, String event, String format, Object... args)302 public void event(Loggable recordEntry, String event, String format, Object... args) { 303 String msg; 304 try { 305 msg = (args == null || args.length == 0) ? format 306 : String.format(Locale.US, format, args); 307 } catch (IllegalFormatException ife) { 308 Log.e(this, ife, "IllegalFormatException: formatString='%s' numArgs=%d", format, 309 args.length); 310 msg = format + " (An error occurred while formatting the message.)"; 311 } 312 313 event(recordEntry, event, msg); 314 } 315 dumpEvents(IndentingPrintWriter pw)316 public void dumpEvents(IndentingPrintWriter pw) { 317 pw.println("Historical Events:"); 318 pw.increaseIndent(); 319 for (EventRecord eventRecord : mEventRecords) { 320 eventRecord.dump(pw); 321 } 322 pw.decreaseIndent(); 323 } 324 325 /** 326 * Dumps events in a timeline format. 327 * @param pw The {@link IndentingPrintWriter} to output the timeline to. 328 * @hide 329 */ dumpEventsTimeline(IndentingPrintWriter pw)330 public void dumpEventsTimeline(IndentingPrintWriter pw) { 331 pw.println("Historical Events (sorted by time):"); 332 333 // Flatten event records out for sorting. 334 List<Pair<Loggable, Event>> events = new ArrayList<>(); 335 for (EventRecord er : mEventRecords) { 336 for (Event ev : er.getEvents()) { 337 events.add(new Pair<>(er.getRecordEntry(), ev)); 338 } 339 } 340 341 // Sort by event time. This might result in out-of-order seeming events if the timezone 342 // changes somewhere in the middle. 343 Comparator<Pair<Loggable, Event>> byEventTime = 344 Comparator.comparingLong(e -> e.second.time); 345 events.sort(byEventTime); 346 347 pw.increaseIndent(); 348 for (Pair<Loggable, Event> event : events) { 349 pw.print(event.second.timestampString); 350 pw.print(","); 351 pw.print(event.first.getId()); 352 pw.print(","); 353 pw.print(event.second.eventId); 354 pw.print(","); 355 pw.println(event.second.data); 356 } 357 pw.decreaseIndent(); 358 } 359 changeEventCacheSize(int newSize)360 public void changeEventCacheSize(int newSize) { 361 362 // Resize the event queue. 363 LinkedBlockingQueue<EventRecord> oldEventLog = mEventRecords; 364 mEventRecords = new LinkedBlockingQueue<>(newSize); 365 mCallEventRecordMap.clear(); 366 367 oldEventLog.forEach((newRecord -> { 368 Loggable recordEntry = newRecord.getRecordEntry(); 369 // Copy the existing queue into the new one. 370 // First remove the oldest entry if no new ones exist. 371 if (mEventRecords.remainingCapacity() == 0) { 372 EventRecord record = mEventRecords.poll(); 373 if (record != null) { 374 mCallEventRecordMap.remove(record.getRecordEntry()); 375 } 376 } 377 378 // Now add a new entry 379 mEventRecords.add(newRecord); 380 mCallEventRecordMap.put(recordEntry, newRecord); 381 382 // Don't worry about notifying mEventListeners, since we are just resizing the records. 383 })); 384 } 385 registerEventListener(EventListener e)386 public void registerEventListener(EventListener e) { 387 if (e != null) { 388 synchronized (mSync) { 389 mEventListeners.add(e); 390 } 391 } 392 } 393 394 @VisibleForTesting getEventRecords()395 public LinkedBlockingQueue<EventRecord> getEventRecords() { 396 return mEventRecords; 397 } 398 399 @VisibleForTesting getCallEventRecordMap()400 public Map<Loggable, EventRecord> getCallEventRecordMap() { 401 return mCallEventRecordMap; 402 } 403 addEventRecord(EventRecord newRecord)404 private void addEventRecord(EventRecord newRecord) { 405 Loggable recordEntry = newRecord.getRecordEntry(); 406 407 // First remove the oldest entry if no new ones exist. 408 if (mEventRecords.remainingCapacity() == 0) { 409 EventRecord record = mEventRecords.poll(); 410 if (record != null) { 411 mCallEventRecordMap.remove(record.getRecordEntry()); 412 } 413 } 414 415 // Now add a new entry 416 mEventRecords.add(newRecord); 417 mCallEventRecordMap.put(recordEntry, newRecord); 418 synchronized (mSync) { 419 for (EventListener l : mEventListeners) { 420 l.eventRecordAdded(newRecord); 421 } 422 } 423 } 424 } 425