blob: dacc20b0fd861fa5b6ea5969131850e653a6ee63 [file] [log] [blame]
/*
* Copyright (C) 2016 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.google.idea.blaze.android.run.runner;
import com.android.ddmlib.IDevice;
import com.android.sdklib.AndroidVersion;
import com.android.tools.idea.run.AndroidLaunchTasksProvider;
import com.android.tools.idea.run.ApkProvisionException;
import com.android.tools.idea.run.ApplicationIdProvider;
import com.android.tools.idea.run.ConsolePrinter;
import com.android.tools.idea.run.LaunchOptions;
import com.android.tools.idea.run.editor.AndroidDebugger;
import com.android.tools.idea.run.editor.AndroidDebuggerState;
import com.android.tools.idea.run.tasks.ClearLogcatTask;
import com.android.tools.idea.run.tasks.DebugConnectorTask;
import com.android.tools.idea.run.tasks.DismissKeyguardTask;
import com.android.tools.idea.run.tasks.LaunchTask;
import com.android.tools.idea.run.tasks.LaunchTasksProvider;
import com.android.tools.idea.run.tasks.ShowLogcatTask;
import com.android.tools.idea.run.util.LaunchStatus;
import com.android.tools.idea.run.util.ProcessHandlerLaunchStatus;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.idea.blaze.android.run.binary.UserIdHelper;
import com.intellij.execution.ExecutionException;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.project.Project;
import java.util.List;
import java.util.Set;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
/** Normal launch tasks provider. */
public class BlazeAndroidLaunchTasksProvider implements LaunchTasksProvider {
private static final Logger LOG = Logger.getInstance(BlazeAndroidLaunchTasksProvider.class);
private final Project project;
private final BlazeAndroidRunContext runContext;
private final ApplicationIdProvider applicationIdProvider;
private final LaunchOptions.Builder launchOptionsBuilder;
private final boolean isDebug;
private final boolean monitorRemoteProcess;
private final BlazeAndroidRunConfigurationDebuggerManager debuggerManager;
public BlazeAndroidLaunchTasksProvider(
Project project,
BlazeAndroidRunContext runContext,
ApplicationIdProvider applicationIdProvider,
LaunchOptions.Builder launchOptionsBuilder,
boolean isDebug,
boolean monitorRemoteProcess,
BlazeAndroidRunConfigurationDebuggerManager debuggerManager) {
this.project = project;
this.runContext = runContext;
this.applicationIdProvider = applicationIdProvider;
this.launchOptionsBuilder = launchOptionsBuilder;
this.isDebug = isDebug;
this.monitorRemoteProcess = monitorRemoteProcess;
this.debuggerManager = debuggerManager;
}
@NotNull
@Override
public List<LaunchTask> getTasks(
@NotNull IDevice device,
@NotNull LaunchStatus launchStatus,
@NotNull ConsolePrinter consolePrinter)
throws ExecutionException {
final List<LaunchTask> launchTasks = Lists.newArrayList();
Integer userId = runContext.getUserId(device, consolePrinter);
launchOptionsBuilder.setPmInstallOptions(UserIdHelper.getFlagsFromUserId(userId));
LaunchOptions launchOptions = launchOptionsBuilder.build();
if (launchOptions.isClearLogcatBeforeStart()) {
launchTasks.add(new ClearLogcatTask(project));
}
launchTasks.add(new DismissKeyguardTask());
if (launchOptions.isDeploy()) {
ImmutableList<LaunchTask> deployTasks = runContext.getDeployTasks(device, launchOptions);
launchTasks.addAll(deployTasks);
}
if (launchStatus.isLaunchTerminated()) {
return launchTasks;
}
String packageName;
try {
packageName = applicationIdProvider.getPackageName();
ProcessHandlerLaunchStatus processHandlerLaunchStatus =
(ProcessHandlerLaunchStatus) launchStatus;
LaunchTask appLaunchTask =
runContext.getApplicationLaunchTask(
launchOptions,
userId,
debuggerManager.getAndroidDebugger(),
debuggerManager.getAndroidDebuggerState(project),
processHandlerLaunchStatus);
if (appLaunchTask != null) {
launchTasks.add(appLaunchTask);
}
} catch (ApkProvisionException e) {
LOG.error(e);
launchStatus.terminateLaunch("Unable to determine application id: " + e);
return ImmutableList.of();
} catch (ExecutionException e) {
launchStatus.terminateLaunch(e.getMessage());
return ImmutableList.of();
}
if (!launchOptions.isDebug() && launchOptions.isOpenLogcatAutomatically()) {
launchTasks.add(new ShowLogcatTask(project, packageName));
}
return launchTasks;
}
@Nullable
@Override
public DebugConnectorTask getConnectDebuggerTask(
@NotNull LaunchStatus launchStatus, @Nullable AndroidVersion version) {
if (!isDebug) {
return null;
}
Set<String> packageIds = Sets.newHashSet();
try {
String packageName = applicationIdProvider.getPackageName();
packageIds.add(packageName);
} catch (ApkProvisionException e) {
Logger.getInstance(AndroidLaunchTasksProvider.class).error(e);
}
try {
String packageName = applicationIdProvider.getTestPackageName();
if (packageName != null) {
packageIds.add(packageName);
}
} catch (ApkProvisionException e) {
// not as severe as failing to obtain package id for main application
Logger.getInstance(AndroidLaunchTasksProvider.class)
.warn(
"Unable to obtain test package name, will not connect debugger "
+ "if tests don't instantiate main application");
}
AndroidDebugger androidDebugger = debuggerManager.getAndroidDebugger();
AndroidDebuggerState androidDebuggerState = debuggerManager.getAndroidDebuggerState(project);
if (androidDebugger == null || androidDebuggerState == null) {
return null;
}
try {
return runContext.getDebuggerTask(
androidDebugger, androidDebuggerState, packageIds, monitorRemoteProcess());
} catch (ExecutionException e) {
launchStatus.terminateLaunch(e.getMessage());
return null;
}
}
@Override
public boolean createsNewProcess() {
return true;
}
@Override
public boolean monitorRemoteProcess() {
return monitorRemoteProcess;
}
}