1 /* 2 * Copyright (C) 2019 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.server.wm.intent; 18 19 import static java.util.stream.Collectors.toList; 20 21 import android.content.ComponentName; 22 import android.content.Intent; 23 import android.net.Uri; 24 import android.server.wm.WindowManagerState; 25 26 import com.google.common.collect.Lists; 27 28 import org.json.JSONArray; 29 import org.json.JSONException; 30 import org.json.JSONObject; 31 32 import java.util.ArrayList; 33 import java.util.Arrays; 34 import java.util.Collections; 35 import java.util.List; 36 import java.util.Map; 37 import java.util.Objects; 38 import java.util.stream.Collectors; 39 import java.util.stream.Stream; 40 41 /** 42 * The intent tests are generated by running a series of intents and then recording the end state 43 * of the system. This class contains all the models needed to store the intents that were used to 44 * create the test case and the end states so that they can be asserted on. 45 * 46 * All test cases are serialized to JSON and stored in a single file per testcase. 47 */ 48 public class Persistence { 49 50 /** 51 * The highest level entity in the JSON file 52 */ 53 public static class TestCase { 54 private static final String SETUP_KEY = "setup"; 55 private static final String INITIAL_STATE_KEY = "initialState"; 56 private static final String END_STATE_KEY = "endState"; 57 58 /** 59 * Contains the {@link android.content.Intent}-s that will be launched in this test case. 60 */ 61 private final Setup mSetup; 62 63 /** 64 * The state of the system after the {@link Setup#mInitialIntents} have been launched. 65 */ 66 private final StateDump mInitialState; 67 68 /** 69 * The state of the system after the {@link Setup#mAct} have been launched 70 */ 71 private final StateDump mEndState; 72 73 /** 74 * The name of the testCase, usually the file name it is stored in. 75 * Not actually persisted to json, since it is only used for presentation purposes. 76 */ 77 private final String mName; 78 TestCase(Setup setup, StateDump initialState, StateDump endState, String name)79 public TestCase(Setup setup, StateDump initialState, 80 StateDump endState, String name) { 81 mSetup = setup; 82 mInitialState = initialState; 83 mEndState = endState; 84 mName = name; 85 } 86 toJson()87 public JSONObject toJson() throws JSONException { 88 return new JSONObject() 89 .put(SETUP_KEY, mSetup.toJson()) 90 .put(INITIAL_STATE_KEY, mInitialState.toJson()) 91 .put(END_STATE_KEY, mEndState.toJson()); 92 } 93 fromJson(JSONObject object, Map<String, IntentFlag> table, String name)94 public static TestCase fromJson(JSONObject object, 95 Map<String, IntentFlag> table, String name) throws JSONException { 96 return new TestCase(Setup.fromJson(object.getJSONObject(SETUP_KEY), table), 97 StateDump.fromJson(object.getJSONObject(INITIAL_STATE_KEY)), 98 StateDump.fromJson(object.getJSONObject(END_STATE_KEY)), name); 99 } 100 getSetup()101 public Setup getSetup() { 102 return mSetup; 103 } 104 getInitialState()105 public StateDump getInitialState() { 106 return mInitialState; 107 } 108 getName()109 public String getName() { 110 return mName; 111 } 112 getEndState()113 public StateDump getEndState() { 114 return mEndState; 115 } 116 } 117 118 /** 119 * Setup consists of two stages. Firstly a list of intents to bring the system in the state we 120 * want to test something in. Secondly a list of intents to bring the system to the final state. 121 */ 122 public static class Setup { 123 private static final String INITIAL_INTENT_KEY = "initialIntents"; 124 private static final String ACT_KEY = "act"; 125 /** 126 * The intent(s) used to bring the system to the initial state. 127 */ 128 private final List<GenerationIntent> mInitialIntents; 129 130 /** 131 * The intent(s) that we actually want to test. 132 */ 133 private final List<GenerationIntent> mAct; 134 Setup(List<GenerationIntent> initialIntents, List<GenerationIntent> act)135 public Setup(List<GenerationIntent> initialIntents, List<GenerationIntent> act) { 136 mInitialIntents = initialIntents; 137 mAct = act; 138 } 139 componentsInCase()140 public List<ComponentName> componentsInCase() { 141 return Stream.concat(mInitialIntents.stream(), mAct.stream()) 142 .map(GenerationIntent::getActualIntent) 143 .map(Intent::getComponent) 144 .collect(Collectors.toList()); 145 } 146 toJson()147 public JSONObject toJson() throws JSONException { 148 return new JSONObject() 149 .put(INITIAL_INTENT_KEY, intentsToJson(mInitialIntents)) 150 .put(ACT_KEY, intentsToJson(mAct)); 151 } 152 fromJson(JSONObject object, Map<String, IntentFlag> table)153 public static Setup fromJson(JSONObject object, 154 Map<String, IntentFlag> table) throws JSONException { 155 List<GenerationIntent> initialState = intentsFromJson( 156 object.getJSONArray(INITIAL_INTENT_KEY), table); 157 List<GenerationIntent> act = intentsFromJson(object.getJSONArray(ACT_KEY), table); 158 159 return new Setup(initialState, act); 160 } 161 162 intentsToJson(List<GenerationIntent> intents)163 public static JSONArray intentsToJson(List<GenerationIntent> intents) 164 throws JSONException { 165 166 JSONArray intentArray = new JSONArray(); 167 for (GenerationIntent intent : intents) { 168 intentArray.put(intent.toJson()); 169 } 170 return intentArray; 171 } 172 intentsFromJson(JSONArray intentArray, Map<String, IntentFlag> table)173 public static List<GenerationIntent> intentsFromJson(JSONArray intentArray, 174 Map<String, IntentFlag> table) throws JSONException { 175 List<GenerationIntent> intents = new ArrayList<>(); 176 177 for (int i = 0; i < intentArray.length(); i++) { 178 JSONObject object = (JSONObject) intentArray.get(i); 179 GenerationIntent intent = GenerationIntent.fromJson(object, table); 180 181 intents.add(intent); 182 } 183 184 return intents; 185 } 186 getInitialIntents()187 public List<GenerationIntent> getInitialIntents() { 188 return mInitialIntents; 189 } 190 getAct()191 public List<GenerationIntent> getAct() { 192 return mAct; 193 } 194 } 195 196 /** 197 * An representation of an {@link android.content.Intent} that can be (de)serialized to / from 198 * JSON. It abstracts whether the context it should be started from is implicitly or explicitly 199 * specified. 200 */ 201 interface GenerationIntent { getActualIntent()202 Intent getActualIntent(); 203 toJson()204 JSONObject toJson() throws JSONException; 205 getLaunchFromIndex(int currentPosition)206 int getLaunchFromIndex(int currentPosition); 207 startForResult()208 boolean startForResult(); 209 fromJson(JSONObject object, Map<String, IntentFlag> table)210 static GenerationIntent fromJson(JSONObject object, Map<String, IntentFlag> table) 211 throws JSONException { 212 if (object.has(LaunchFromIntent.LAUNCH_FROM_KEY)) { 213 return LaunchFromIntent.fromJson(object, table); 214 } else { 215 return LaunchIntent.fromJson(object, table); 216 } 217 } 218 } 219 220 /** 221 * Representation of {@link android.content.Intent} used by the {@link LaunchSequence} api. 222 * It be can used to normally start activities, to start activities for result and Intent Flags 223 * can be added using {@link LaunchIntent#withFlags(IntentFlag...)} 224 */ 225 static class LaunchIntent implements GenerationIntent { 226 private static final String FLAGS_KEY = "flags"; 227 private static final String PACKAGE_KEY = "package"; 228 private static final String CLASS_KEY = "class"; 229 private static final String DATA_KEY = "data"; 230 private static final String START_FOR_RESULT_KEY = "startForResult"; 231 232 private final List<IntentFlag> mIntentFlags; 233 private final ComponentName mComponentName; 234 private final String mData; 235 private final boolean mStartForResult; 236 LaunchIntent(List<IntentFlag> intentFlags, ComponentName componentName, String data, boolean startForResult)237 public LaunchIntent(List<IntentFlag> intentFlags, ComponentName componentName, String data, 238 boolean startForResult) { 239 mIntentFlags = intentFlags; 240 mComponentName = componentName; 241 mData = data; 242 mStartForResult = startForResult; 243 } 244 245 @Override getActualIntent()246 public Intent getActualIntent() { 247 final Intent intent = new Intent().setComponent(mComponentName).setFlags(buildFlag()); 248 if (mData != null && !mData.isEmpty()) { 249 intent.setData(Uri.parse(mData)); 250 } 251 return intent; 252 } 253 254 @Override getLaunchFromIndex(int currentPosition)255 public int getLaunchFromIndex(int currentPosition) { 256 return currentPosition - 1; 257 } 258 259 @Override startForResult()260 public boolean startForResult() { 261 return mStartForResult; 262 } 263 buildFlag()264 public int buildFlag() { 265 int flag = 0; 266 for (IntentFlag intentFlag : mIntentFlags) { 267 flag |= intentFlag.flag; 268 } 269 270 return flag; 271 } 272 humanReadableFlags()273 public String humanReadableFlags() { 274 return mIntentFlags.stream().map(IntentFlag::toString).collect( 275 Collectors.joining(" | ")); 276 } 277 fromJson(JSONObject fakeIntent, Map<String, IntentFlag> table)278 public static LaunchIntent fromJson(JSONObject fakeIntent, Map<String, IntentFlag> table) 279 throws JSONException { 280 List<IntentFlag> flags = IntentFlag.parse(table, fakeIntent.getString(FLAGS_KEY)); 281 282 boolean startForResult = fakeIntent.optBoolean(START_FOR_RESULT_KEY, false); 283 String uri = fakeIntent.optString(DATA_KEY); 284 return new LaunchIntent(flags, 285 new ComponentName( 286 fakeIntent.getString(PACKAGE_KEY), 287 fakeIntent.getString(CLASS_KEY)), 288 uri, 289 startForResult); 290 } 291 292 @Override toJson()293 public JSONObject toJson() throws JSONException { 294 return new JSONObject().put(FLAGS_KEY, this.humanReadableFlags()) 295 .put(CLASS_KEY, this.mComponentName.getClassName()) 296 .put(PACKAGE_KEY, this.mComponentName.getPackageName()) 297 .put(START_FOR_RESULT_KEY, mStartForResult); 298 } 299 withFlags(IntentFlag... flags)300 public LaunchIntent withFlags(IntentFlag... flags) { 301 List<IntentFlag> intentFlags = Lists.newArrayList(mIntentFlags); 302 Collections.addAll(intentFlags, flags); 303 return new LaunchIntent(intentFlags, mComponentName, mData, mStartForResult); 304 } 305 getIntentFlags()306 public List<IntentFlag> getIntentFlags() { 307 return mIntentFlags; 308 } 309 getComponentName()310 public ComponentName getComponentName() { 311 return mComponentName; 312 } 313 } 314 315 /** 316 * Representation of {@link android.content.Intent} used by the {@link LaunchSequence} api. 317 * It can used to normally start activities, to start activities for result and Intent Flags 318 * can 319 * be added using {@link LaunchIntent#withFlags(IntentFlag...)} just like {@link LaunchIntent} 320 * 321 * However {@link LaunchFromIntent} also supports launching from a activity earlier in the 322 * launch sequence. This can be done using {@link LaunchSequence#act} and related methods. 323 */ 324 static class LaunchFromIntent implements GenerationIntent { 325 static final String LAUNCH_FROM_KEY = "launchFrom"; 326 327 /** 328 * The underlying {@link LaunchIntent} that we are wrapping with the launch point behaviour. 329 */ 330 private final LaunchIntent mLaunchIntent; 331 332 /** 333 * The index in the activityLog maintained by {@link LaunchRunner}, used to retrieve the 334 * activity from the log to start this {@link LaunchIntent} from. 335 */ 336 private final int mLaunchFrom; 337 LaunchFromIntent(LaunchIntent fakeIntent, int launchFrom)338 LaunchFromIntent(LaunchIntent fakeIntent, int launchFrom) { 339 mLaunchIntent = fakeIntent; 340 mLaunchFrom = launchFrom; 341 } 342 343 344 @Override getActualIntent()345 public Intent getActualIntent() { 346 return mLaunchIntent.getActualIntent(); 347 } 348 349 @Override getLaunchFromIndex(int currentPosition)350 public int getLaunchFromIndex(int currentPosition) { 351 return mLaunchFrom; 352 } 353 354 @Override startForResult()355 public boolean startForResult() { 356 return mLaunchIntent.mStartForResult; 357 } 358 359 @Override toJson()360 public JSONObject toJson() throws JSONException { 361 return mLaunchIntent.toJson() 362 .put(LAUNCH_FROM_KEY, mLaunchFrom); 363 } 364 fromJson(JSONObject object, Map<String, IntentFlag> table)365 public static LaunchFromIntent fromJson(JSONObject object, Map<String, IntentFlag> table) 366 throws JSONException { 367 LaunchIntent fakeIntent = LaunchIntent.fromJson(object, table); 368 int launchFrom = object.optInt(LAUNCH_FROM_KEY, -1); 369 370 return new LaunchFromIntent(fakeIntent, launchFrom); 371 } 372 prepareSerialisation(List<LaunchFromIntent> intents)373 static List<GenerationIntent> prepareSerialisation(List<LaunchFromIntent> intents) { 374 return prepareSerialisation(intents, 0); 375 } 376 377 // In serialized form we only want to store the launch from index if it deviates from the 378 // default, the default being the previous activity. prepareSerialisation(List<LaunchFromIntent> intents, int base)379 static List<GenerationIntent> prepareSerialisation(List<LaunchFromIntent> intents, 380 int base) { 381 List<GenerationIntent> serializeIntents = Lists.newArrayList(); 382 for (int i = 0; i < intents.size(); i++) { 383 LaunchFromIntent launchFromIntent = intents.get(i); 384 serializeIntents.add(launchFromIntent.forget(base + i)); 385 } 386 387 return serializeIntents; 388 } 389 forget(int currentIndex)390 public GenerationIntent forget(int currentIndex) { 391 if (mLaunchFrom == currentIndex - 1) { 392 return this.mLaunchIntent; 393 } else { 394 return this; 395 } 396 } 397 getLaunchFrom()398 public int getLaunchFrom() { 399 return mLaunchFrom; 400 } 401 } 402 403 /** 404 * An intent flag that also stores the name of the flag. 405 * It is used to be able to put the flags in human readable form in the JSON file. 406 */ 407 static class IntentFlag { 408 /** 409 * The underlying flag, should be a value from Intent.FLAG_ACTIVITY_*. 410 */ 411 public final int flag; 412 /** 413 * The name of the flag. 414 */ 415 public final String name; 416 IntentFlag(int flag, String name)417 public IntentFlag(int flag, String name) { 418 this.flag = flag; 419 this.name = name; 420 } 421 getFlag()422 public int getFlag() { 423 return flag; 424 } 425 getName()426 public String getName() { 427 return name; 428 } 429 combine(IntentFlag other)430 public int combine(IntentFlag other) { 431 return other.flag | flag; 432 } 433 parse(Map<String, IntentFlag> names, String flagsToParse)434 public static List<IntentFlag> parse(Map<String, IntentFlag> names, String flagsToParse) { 435 String[] split = flagsToParse.replaceAll("\\s", "").split("\\|"); 436 return Arrays.stream(split).map(names::get).collect(toList()); 437 } 438 toString()439 public String toString() { 440 return name; 441 } 442 } 443 flag(int flag, String name)444 static IntentFlag flag(int flag, String name) { 445 return new IntentFlag(flag, name); 446 } 447 448 public static class StateDump { 449 private static final String TASKS_KEY = "tasks"; 450 451 /** 452 * The Tasks in this stack ordered from most recent to least recent. 453 */ 454 private final List<TaskState> mTasks; 455 fromTasks(List<WindowManagerState.Task> activityTasks, List<WindowManagerState.Task> baseStacks)456 public static StateDump fromTasks(List<WindowManagerState.Task> activityTasks, 457 List<WindowManagerState.Task> baseStacks) { 458 List<TaskState> tasks = new ArrayList<>(); 459 for (WindowManagerState.Task task : trimTasks(activityTasks, baseStacks)) { 460 tasks.add(new TaskState(task)); 461 } 462 return new StateDump(tasks); 463 } 464 StateDump(List<TaskState> tasks)465 private StateDump(List<TaskState> tasks) { 466 mTasks = tasks; 467 } 468 toJson()469 JSONObject toJson() throws JSONException { 470 JSONArray tasks = new JSONArray(); 471 for (TaskState task : mTasks) { 472 tasks.put(task.toJson()); 473 } 474 475 return new JSONObject().put(TASKS_KEY, tasks); 476 } 477 fromJson(JSONObject object)478 static StateDump fromJson(JSONObject object) throws JSONException { 479 JSONArray jsonTasks = object.getJSONArray(TASKS_KEY); 480 List<TaskState> tasks = new ArrayList<>(); 481 482 for (int i = 0; i < jsonTasks.length(); i++) { 483 tasks.add(TaskState.fromJson((JSONObject) jsonTasks.get(i))); 484 } 485 486 return new StateDump(tasks); 487 } 488 489 /** 490 * To make the state dump non device specific we remove every task that was present 491 * in the system before recording, by their ID. For example a task containing the launcher 492 * activity. 493 */ trimTasks( List<WindowManagerState.Task> toTrim, List<WindowManagerState.Task> trimFrom)494 public static List<WindowManagerState.Task> trimTasks( 495 List<WindowManagerState.Task> toTrim, 496 List<WindowManagerState.Task> trimFrom) { 497 498 for (WindowManagerState.Task task : trimFrom) { 499 toTrim.removeIf(t -> t.getRootTaskId() == task.getRootTaskId()); 500 } 501 502 return toTrim; 503 } 504 505 @Override equals(Object o)506 public boolean equals(Object o) { 507 if (this == o) return true; 508 if (o == null || getClass() != o.getClass()) return false; 509 StateDump stateDump = (StateDump) o; 510 return Objects.equals(mTasks, stateDump.mTasks); 511 } 512 513 @Override hashCode()514 public int hashCode() { 515 return Objects.hash(mTasks); 516 } 517 } 518 519 public static class TaskState { 520 521 private static final String STATE_RESUMED = "RESUMED"; 522 private static final String ACTIVITIES_KEY = "activities"; 523 524 /** 525 * The component name of the resumedActivity in this state, empty string if there is none. 526 */ 527 private final String mResumedActivity; 528 529 /** 530 * The activities in this task ordered from most recent to least recent. 531 */ 532 private final List<ActivityState> mActivities = new ArrayList<>(); 533 TaskState(JSONArray jsonActivities)534 private TaskState(JSONArray jsonActivities) throws JSONException { 535 String resumedActivity = ""; 536 for (int i = 0; i < jsonActivities.length(); i++) { 537 final ActivityState activity = 538 ActivityState.fromJson((JSONObject) jsonActivities.get(i)); 539 // The json file shouldn't define multiple resumed activities, but it is fine that 540 // the test will fail when comparing to the real state. 541 if (STATE_RESUMED.equals(activity.getState())) { 542 resumedActivity = activity.getName(); 543 } 544 mActivities.add(activity); 545 } 546 547 mResumedActivity = resumedActivity; 548 } 549 TaskState(WindowManagerState.Task state)550 public TaskState(WindowManagerState.Task state) { 551 final String resumedActivity = state.getResumedActivity(); 552 mResumedActivity = resumedActivity != null ? resumedActivity : ""; 553 for (WindowManagerState.Activity activity : state.getActivities()) { 554 this.mActivities.add(new ActivityState(activity)); 555 } 556 } 557 toJson()558 JSONObject toJson() throws JSONException { 559 JSONArray jsonActivities = new JSONArray(); 560 561 for (ActivityState activity : mActivities) { 562 jsonActivities.put(activity.toJson()); 563 } 564 565 return new JSONObject() 566 .put(ACTIVITIES_KEY, jsonActivities); 567 } 568 fromJson(JSONObject object)569 static TaskState fromJson(JSONObject object) throws JSONException { 570 return new TaskState(object.getJSONArray(ACTIVITIES_KEY)); 571 } 572 getActivities()573 public List<ActivityState> getActivities() { 574 return mActivities; 575 } 576 577 @Override equals(Object o)578 public boolean equals(Object o) { 579 if (this == o) return true; 580 if (o == null || getClass() != o.getClass()) return false; 581 TaskState task = (TaskState) o; 582 return Objects.equals(mResumedActivity, task.mResumedActivity) 583 && Objects.equals(mActivities, task.mActivities); 584 } 585 586 @Override hashCode()587 public int hashCode() { 588 return Objects.hash(mResumedActivity, mActivities); 589 } 590 } 591 592 public static class ActivityState { 593 private static final String NAME_KEY = "name"; 594 private static final String STATE_KEY = "state"; 595 /** 596 * The componentName of this activity. 597 */ 598 private final String mComponentName; 599 600 /** 601 * The lifecycle state this activity is in. 602 */ 603 private final String mLifeCycleState; 604 ActivityState(String name, String state)605 public ActivityState(String name, String state) { 606 mComponentName = name; 607 mLifeCycleState = state; 608 } 609 ActivityState(WindowManagerState.Activity activity)610 public ActivityState(WindowManagerState.Activity activity) { 611 mComponentName = activity.getName(); 612 mLifeCycleState = activity.getState(); 613 } 614 615 toJson()616 JSONObject toJson() throws JSONException { 617 return new JSONObject().put(NAME_KEY, mComponentName).put(STATE_KEY, mLifeCycleState); 618 } 619 fromJson(JSONObject object)620 static ActivityState fromJson(JSONObject object) throws JSONException { 621 return new ActivityState(object.getString(NAME_KEY), object.getString(STATE_KEY)); 622 } 623 624 @Override equals(Object o)625 public boolean equals(Object o) { 626 if (this == o) return true; 627 if (o == null || getClass() != o.getClass()) return false; 628 ActivityState activity = (ActivityState) o; 629 return Objects.equals(mComponentName, activity.mComponentName) && 630 Objects.equals(mLifeCycleState, activity.mLifeCycleState); 631 } 632 633 @Override hashCode()634 public int hashCode() { 635 return Objects.hash(mComponentName, mLifeCycleState); 636 } 637 getName()638 public String getName() { 639 return mComponentName; 640 } 641 getState()642 public String getState() { 643 return mLifeCycleState; 644 } 645 } 646 } 647