blob: 4521b6fa595199473fff022158ed292feecd5c07 [file] [log] [blame]
/*
* Copyright 2022 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.tests.odsign;
import static com.google.common.truth.Truth.assertWithMessage;
import static org.junit.Assert.fail;
import static org.junit.Assume.assumeTrue;
import com.android.tradefed.device.DeviceNotAvailableException;
import com.android.tradefed.device.ITestDevice;
import com.android.tradefed.device.TestDevice;
import com.android.tradefed.util.CommandResult;
import java.util.concurrent.TimeUnit;
public class CompOsTestUtils {
public static final String PENDING_ARTIFACTS_DIR =
"/data/misc/apexdata/com.android.art/compos-pending";
/** Maximum time for a slow VM like cuttlefish to boot and finish odrefresh. */
// odrefresh overall timeout is currently 480s; add some generous padding for VM startup.
private static final int VM_ODREFRESH_MAX_SECONDS = 480 + 60;
/** Waiting time for the job to be scheduled after staging an APEX */
private static final int JOB_CREATION_MAX_SECONDS = 5;
/** Waiting time before starting to check odrefresh progress. */
private static final int SECONDS_BEFORE_PROGRESS_CHECK = 30;
/** Job ID of the pending compilation with staged APEXes. */
private static final String JOB_ID = "5132251";
private final ITestDevice mDevice;
public CompOsTestUtils(ITestDevice device) {
mDevice = device;
}
/**
* Start CompOS compilation right away, and return once the job completes successfully.
*
* <p>Once the test APK is installed, a CompilationJob is (asynchronously) scheduled to run
* when certain criteria are met, e.g. the device is charging and idle. Since we don't want
* to wait in the test, here we start the job by ID as soon as it is scheduled.
*/
public void runCompilationJobEarlyAndWait() throws Exception {
waitForJobToBeScheduled();
assertCommandSucceeds("cmd jobscheduler run -f android " + JOB_ID);
// It takes time. Just don't spam.
TimeUnit.SECONDS.sleep(SECONDS_BEFORE_PROGRESS_CHECK);
// The job runs asynchronously. To wait until it completes.
waitForJobExit(VM_ODREFRESH_MAX_SECONDS - SECONDS_BEFORE_PROGRESS_CHECK);
}
public String checksumDirectoryContentPartial(String path) throws Exception {
// Sort by filename (second column) to make comparison easier.
// Filter out compos.info* (which will be deleted at boot) and cache-info.xml
// compos.info.signature since it's only generated by CompOS.
return assertCommandSucceeds("cd " + path + " && find -type f -exec sha256sum {} \\;"
+ "| grep -v cache-info.xml | grep -v compos.info"
+ "| sort -k2");
}
private void waitForJobToBeScheduled()
throws Exception {
for (int i = 0; i < JOB_CREATION_MAX_SECONDS; i++) {
CommandResult result = mDevice.executeShellV2Command(
"cmd jobscheduler get-job-state android " + JOB_ID);
String state = result.getStdout();
if (state.startsWith("unknown")) {
// The job hasn't been scheduled yet. So try again.
TimeUnit.SECONDS.sleep(1);
} else if (result.getExitCode() != 0) {
fail("Failing due to unexpected cmd jobscheduler exit code: " + result);
} else {
// The job exists, which is all we care about here
return;
}
}
fail("Timed out waiting for the job to be scheduled");
}
private void waitForJobExit(int timeout) throws Exception {
for (int i = 0; i < timeout; i++) {
CommandResult result = mDevice.executeShellV2Command(
"cmd jobscheduler get-job-state android " + JOB_ID);
String state = result.getStdout();
if (state.contains("ready") || state.contains("active")) {
TimeUnit.SECONDS.sleep(1);
} else if (state.startsWith("unknown")) {
// Job has completed
return;
} else {
String context = (result.getExitCode() == 0) ? result.getStdout()
: result.toString();
fail("Failing due to unexpected job state: " + context);
}
}
fail("Timed out waiting for the job to complete");
}
public void assumeCompOsPresent() throws Exception {
// We have to have kernel support for a VM.
assumeTrue("Need an actual TestDevice", mDevice instanceof TestDevice);
TestDevice testDevice = (TestDevice) mDevice;
assumeTrue("Requires VM support", testDevice.supportsMicrodroid());
// And the CompOS APEX must be present.
assumeTrue(mDevice.doesFileExist("/apex/com.android.compos/"));
}
public void assumeNotOnCuttlefish() throws Exception {
String product = mDevice.getProperty("ro.build.product");
assumeTrue(product != null && !product.startsWith("vsoc_"));
}
private String assertCommandSucceeds(String command) throws DeviceNotAvailableException {
CommandResult result = mDevice.executeShellV2Command(command);
assertWithMessage(result.toString()).that(result.getExitCode()).isEqualTo(0);
return result.getStdout().trim();
}
}