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.os.Parcel; 21 import android.os.Parcelable; 22 import android.telecom.Log; 23 import android.text.TextUtils; 24 25 import com.android.internal.annotations.VisibleForTesting; 26 27 import java.util.ArrayList; 28 29 /** 30 * Stores information about a thread's point of entry into that should persist until that thread 31 * exits. 32 * @hide 33 */ 34 public class Session { 35 36 public static final String START_SESSION = "START_SESSION"; 37 public static final String START_EXTERNAL_SESSION = "START_EXTERNAL_SESSION"; 38 public static final String CREATE_SUBSESSION = "CREATE_SUBSESSION"; 39 public static final String CONTINUE_SUBSESSION = "CONTINUE_SUBSESSION"; 40 public static final String END_SUBSESSION = "END_SUBSESSION"; 41 public static final String END_SESSION = "END_SESSION"; 42 43 public static final String SUBSESSION_SEPARATION_CHAR = "->"; 44 public static final String SESSION_SEPARATION_CHAR_CHILD = "_"; 45 public static final String EXTERNAL_INDICATOR = "E-"; 46 public static final String TRUNCATE_STRING = "..."; 47 48 /** 49 * Initial value of mExecutionEndTimeMs and the final value of {@link #getLocalExecutionTime()} 50 * if the Session is canceled. 51 */ 52 public static final int UNDEFINED = -1; 53 54 public static class Info implements Parcelable { 55 public final String sessionId; 56 public final String methodPath; 57 Info(String id, String path)58 private Info(String id, String path) { 59 sessionId = id; 60 methodPath = path; 61 } 62 getInfo(Session s)63 public static Info getInfo (Session s) { 64 // Create Info based on the truncated method path if the session is external, so we do 65 // not get multiple stacking external sessions (unless we have DEBUG level logging or 66 // lower). 67 return new Info(s.getFullSessionId(), s.getFullMethodPath( 68 !Log.DEBUG && s.isSessionExternal())); 69 } 70 71 /** Responsible for creating Info objects for deserialized Parcels. */ 72 public static final @android.annotation.NonNull Parcelable.Creator<Info> CREATOR = 73 new Parcelable.Creator<Info> () { 74 @Override 75 public Info createFromParcel(Parcel source) { 76 String id = source.readString(); 77 String methodName = source.readString(); 78 return new Info(id, methodName); 79 } 80 81 @Override 82 public Info[] newArray(int size) { 83 return new Info[size]; 84 } 85 }; 86 87 /** {@inheritDoc} */ 88 @Override describeContents()89 public int describeContents() { 90 return 0; 91 } 92 93 /** Writes Info object into a Parcel. */ 94 @Override writeToParcel(Parcel destination, int flags)95 public void writeToParcel(Parcel destination, int flags) { 96 destination.writeString(sessionId); 97 destination.writeString(methodPath); 98 } 99 } 100 101 private String mSessionId; 102 private String mShortMethodName; 103 private long mExecutionStartTimeMs; 104 private long mExecutionEndTimeMs = UNDEFINED; 105 private Session mParentSession; 106 private ArrayList<Session> mChildSessions; 107 private boolean mIsCompleted = false; 108 private boolean mIsExternal = false; 109 private int mChildCounter = 0; 110 // True if this is a subsession that has been started from the same thread as the parent 111 // session. This can happen if Log.startSession(...) is called multiple times on the same 112 // thread in the case of one Telecom entry point method calling another entry point method. 113 // In this case, we can just make this subsession "invisible," but still keep track of it so 114 // that the Log.endSession() calls match up. 115 private boolean mIsStartedFromActiveSession = false; 116 // Optionally provided info about the method/class/component that started the session in order 117 // to make Logging easier. This info will be provided in parentheses along with the session. 118 private String mOwnerInfo; 119 // Cache Full Method path so that recursive population of the full method path only needs to 120 // be calculated once. 121 private String mFullMethodPathCache; 122 Session(String sessionId, String shortMethodName, long startTimeMs, boolean isStartedFromActiveSession, String ownerInfo)123 public Session(String sessionId, String shortMethodName, long startTimeMs, 124 boolean isStartedFromActiveSession, String ownerInfo) { 125 setSessionId(sessionId); 126 setShortMethodName(shortMethodName); 127 mExecutionStartTimeMs = startTimeMs; 128 mParentSession = null; 129 mChildSessions = new ArrayList<>(5); 130 mIsStartedFromActiveSession = isStartedFromActiveSession; 131 mOwnerInfo = ownerInfo; 132 } 133 setSessionId(@onNull String sessionId)134 public void setSessionId(@NonNull String sessionId) { 135 if (sessionId == null) { 136 mSessionId = "?"; 137 } 138 mSessionId = sessionId; 139 } 140 getShortMethodName()141 public String getShortMethodName() { 142 return mShortMethodName; 143 } 144 setShortMethodName(String shortMethodName)145 public void setShortMethodName(String shortMethodName) { 146 if (shortMethodName == null) { 147 shortMethodName = ""; 148 } 149 mShortMethodName = shortMethodName; 150 } 151 setIsExternal(boolean isExternal)152 public void setIsExternal(boolean isExternal) { 153 mIsExternal = isExternal; 154 } 155 isExternal()156 public boolean isExternal() { 157 return mIsExternal; 158 } 159 setParentSession(Session parentSession)160 public void setParentSession(Session parentSession) { 161 mParentSession = parentSession; 162 } 163 addChild(Session childSession)164 public void addChild(Session childSession) { 165 if (childSession != null) { 166 mChildSessions.add(childSession); 167 } 168 } 169 removeChild(Session child)170 public void removeChild(Session child) { 171 if (child != null) { 172 mChildSessions.remove(child); 173 } 174 } 175 getExecutionStartTimeMilliseconds()176 public long getExecutionStartTimeMilliseconds() { 177 return mExecutionStartTimeMs; 178 } 179 setExecutionStartTimeMs(long startTimeMs)180 public void setExecutionStartTimeMs(long startTimeMs) { 181 mExecutionStartTimeMs = startTimeMs; 182 } 183 getParentSession()184 public Session getParentSession() { 185 return mParentSession; 186 } 187 getChildSessions()188 public ArrayList<Session> getChildSessions() { 189 return mChildSessions; 190 } 191 isSessionCompleted()192 public boolean isSessionCompleted() { 193 return mIsCompleted; 194 } 195 isStartedFromActiveSession()196 public boolean isStartedFromActiveSession() { 197 return mIsStartedFromActiveSession; 198 } 199 getInfo()200 public Info getInfo() { 201 return Info.getInfo(this); 202 } 203 204 @VisibleForTesting getSessionId()205 public String getSessionId() { 206 return mSessionId; 207 } 208 209 // Mark this session complete. This will be deleted by Log when all subsessions are complete 210 // as well. markSessionCompleted(long executionEndTimeMs)211 public void markSessionCompleted(long executionEndTimeMs) { 212 mExecutionEndTimeMs = executionEndTimeMs; 213 mIsCompleted = true; 214 } 215 getLocalExecutionTime()216 public long getLocalExecutionTime() { 217 if (mExecutionEndTimeMs == UNDEFINED) { 218 return UNDEFINED; 219 } 220 return mExecutionEndTimeMs - mExecutionStartTimeMs; 221 } 222 getNextChildId()223 public synchronized String getNextChildId() { 224 return String.valueOf(mChildCounter++); 225 } 226 227 // Builds full session id recursively getFullSessionId()228 private String getFullSessionId() { 229 // Cache mParentSession locally to prevent a concurrency problem where 230 // Log.endParentSessions() is called while a logging statement is running (Log.i, for 231 // example) and setting mParentSession to null in a different thread after the null check 232 // occurred. 233 Session parentSession = mParentSession; 234 if (parentSession == null) { 235 return mSessionId; 236 } else { 237 if (Log.VERBOSE) { 238 return parentSession.getFullSessionId() + 239 // Append "_X" to subsession to show subsession designation. 240 SESSION_SEPARATION_CHAR_CHILD + mSessionId; 241 } else { 242 // Only worry about the base ID at the top of the tree. 243 return parentSession.getFullSessionId(); 244 } 245 246 } 247 } 248 249 // Print out the full Session tree from any subsession node printFullSessionTree()250 public String printFullSessionTree() { 251 // Get to the top of the tree 252 Session topNode = this; 253 while (topNode.getParentSession() != null) { 254 topNode = topNode.getParentSession(); 255 } 256 return topNode.printSessionTree(); 257 } 258 259 // Recursively move down session tree using DFS, but print out each node when it is reached. printSessionTree()260 public String printSessionTree() { 261 StringBuilder sb = new StringBuilder(); 262 printSessionTree(0, sb); 263 return sb.toString(); 264 } 265 printSessionTree(int tabI, StringBuilder sb)266 private void printSessionTree(int tabI, StringBuilder sb) { 267 sb.append(toString()); 268 for (Session child : mChildSessions) { 269 sb.append("\n"); 270 for (int i = 0; i <= tabI; i++) { 271 sb.append("\t"); 272 } 273 child.printSessionTree(tabI + 1, sb); 274 } 275 } 276 277 // Recursively concatenate mShortMethodName with the parent Sessions to create full method 278 // path. if truncatePath is set to true, all other external sessions (except for the most 279 // recent) will be truncated to "..." getFullMethodPath(boolean truncatePath)280 public String getFullMethodPath(boolean truncatePath) { 281 StringBuilder sb = new StringBuilder(); 282 getFullMethodPath(sb, truncatePath); 283 return sb.toString(); 284 } 285 getFullMethodPath(StringBuilder sb, boolean truncatePath)286 private synchronized void getFullMethodPath(StringBuilder sb, boolean truncatePath) { 287 // Return cached value for method path. When returning the truncated path, recalculate the 288 // full path without using the cached value. 289 if (!TextUtils.isEmpty(mFullMethodPathCache) && !truncatePath) { 290 sb.append(mFullMethodPathCache); 291 return; 292 } 293 Session parentSession = getParentSession(); 294 boolean isSessionStarted = false; 295 if (parentSession != null) { 296 // Check to see if the session has been renamed yet. If it has not, then the session 297 // has not been continued. 298 isSessionStarted = !mShortMethodName.equals(parentSession.mShortMethodName); 299 parentSession.getFullMethodPath(sb, truncatePath); 300 sb.append(SUBSESSION_SEPARATION_CHAR); 301 } 302 // Encapsulate the external session's method name so it is obvious what part of the session 303 // is external or truncate it if we do not want the entire history. 304 if (isExternal()) { 305 if (truncatePath) { 306 sb.append(TRUNCATE_STRING); 307 } else { 308 sb.append("("); 309 sb.append(mShortMethodName); 310 sb.append(")"); 311 } 312 } else { 313 sb.append(mShortMethodName); 314 } 315 // If we are returning the truncated path, do not save that path as the full path. 316 if (isSessionStarted && !truncatePath) { 317 // Cache this value so that we do not have to do this work next time! 318 // We do not cache the value if the session being evaluated hasn't been continued yet. 319 mFullMethodPathCache = sb.toString(); 320 } 321 } 322 // Recursively move to the top of the tree to see if the parent session is external. isSessionExternal()323 private boolean isSessionExternal() { 324 if (getParentSession() == null) { 325 return isExternal(); 326 } else { 327 return getParentSession().isSessionExternal(); 328 } 329 } 330 331 @Override hashCode()332 public int hashCode() { 333 int result = mSessionId != null ? mSessionId.hashCode() : 0; 334 result = 31 * result + (mShortMethodName != null ? mShortMethodName.hashCode() : 0); 335 result = 31 * result + (int) (mExecutionStartTimeMs ^ (mExecutionStartTimeMs >>> 32)); 336 result = 31 * result + (int) (mExecutionEndTimeMs ^ (mExecutionEndTimeMs >>> 32)); 337 result = 31 * result + (mParentSession != null ? mParentSession.hashCode() : 0); 338 result = 31 * result + (mChildSessions != null ? mChildSessions.hashCode() : 0); 339 result = 31 * result + (mIsCompleted ? 1 : 0); 340 result = 31 * result + mChildCounter; 341 result = 31 * result + (mIsStartedFromActiveSession ? 1 : 0); 342 result = 31 * result + (mOwnerInfo != null ? mOwnerInfo.hashCode() : 0); 343 return result; 344 } 345 346 @Override equals(Object o)347 public boolean equals(Object o) { 348 if (this == o) return true; 349 if (o == null || getClass() != o.getClass()) return false; 350 351 Session session = (Session) o; 352 353 if (mExecutionStartTimeMs != session.mExecutionStartTimeMs) return false; 354 if (mExecutionEndTimeMs != session.mExecutionEndTimeMs) return false; 355 if (mIsCompleted != session.mIsCompleted) return false; 356 if (mChildCounter != session.mChildCounter) return false; 357 if (mIsStartedFromActiveSession != session.mIsStartedFromActiveSession) return false; 358 if (mSessionId != null ? 359 !mSessionId.equals(session.mSessionId) : session.mSessionId != null) 360 return false; 361 if (mShortMethodName != null ? !mShortMethodName.equals(session.mShortMethodName) 362 : session.mShortMethodName != null) 363 return false; 364 if (mParentSession != null ? !mParentSession.equals(session.mParentSession) 365 : session.mParentSession != null) 366 return false; 367 if (mChildSessions != null ? !mChildSessions.equals(session.mChildSessions) 368 : session.mChildSessions != null) 369 return false; 370 return mOwnerInfo != null ? mOwnerInfo.equals(session.mOwnerInfo) 371 : session.mOwnerInfo == null; 372 373 } 374 375 @Override toString()376 public String toString() { 377 if (mParentSession != null && mIsStartedFromActiveSession) { 378 // Log.startSession was called from within another active session. Use the parent's 379 // Id instead of the child to reduce confusion. 380 return mParentSession.toString(); 381 } else { 382 StringBuilder methodName = new StringBuilder(); 383 methodName.append(getFullMethodPath(false /*truncatePath*/)); 384 if (mOwnerInfo != null && !mOwnerInfo.isEmpty()) { 385 methodName.append("(InCall package: "); 386 methodName.append(mOwnerInfo); 387 methodName.append(")"); 388 } 389 return methodName.toString() + "@" + getFullSessionId(); 390 } 391 } 392 } 393