/*
* Copyright 2000-2015 JetBrains s.r.o.
*
* 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.intellij.execution;
import com.intellij.CommonBundle;
import com.intellij.execution.configurations.GeneralCommandLine;
import com.intellij.execution.process.CapturingProcessHandler;
import com.intellij.execution.process.ProcessOutput;
import com.intellij.notification.*;
import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.options.ShowSettingsUtil;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.ui.Messages;
import com.intellij.openapi.util.registry.Registry;
import com.intellij.openapi.vfs.CharsetToolkit;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
import javax.swing.event.HyperlinkEvent;
import java.awt.*;
import java.util.Collections;
import java.util.List;
import static com.intellij.notification.NotificationDisplayType.STICKY_BALLOON;
/**
* Validates the given external executable. If it is not valid, shows notification to fix it.
* Notification group is registered as a {@link STICKY_BALLOON} by default.
*
* @author Kirill Likhodedov
*/
public abstract class ExecutableValidator {
public static final int TIMEOUT_MS = Registry.intValue("vcs.executable.validator.timeout.sec", 60) * 1000;
private static final Logger LOG = Logger.getInstance(ExecutableValidator.class);
private static final NotificationGroup ourNotificationGroup = new NotificationGroup("External Executable Critical Failures", STICKY_BALLOON, true);
@NotNull protected final Project myProject;
@NotNull protected final NotificationsManager myNotificationManager;
@NotNull private final String myNotificationErrorTitle;
@NotNull private final String myNotificationErrorDescription;
/**
* Configures notification and dialog by setting text messages and titles specific to the whoever uses the validator.
*
* @param notificationErrorTitle title of the notification about not valid executable.
* @param notificationErrorDescription description of this notification with a link to fix it (link action is defined by
* {@link #showSettingsAndExpireIfFixed(com.intellij.notification.Notification)}
*/
public ExecutableValidator(@NotNull Project project, @NotNull String notificationErrorTitle, @NotNull String notificationErrorDescription) {
myProject = project;
myNotificationErrorTitle = notificationErrorTitle;
myNotificationErrorDescription = notificationErrorDescription;
myNotificationManager = NotificationsManager.getNotificationsManager();
}
protected abstract String getCurrentExecutable();
/**
* @return the settings configurable display name, where the executable is shown and can be fixed.
* This configurable will be opened if user presses "Fix" on the notification about invalid executable.
*/
@NotNull
protected abstract String getConfigurableDisplayName();
@Nullable
protected Notification validate(@NotNull String executable) {
return !isExecutableValid(executable) ? createDefaultNotification() : null;
}
@NotNull
protected ExecutableNotValidNotification createDefaultNotification() {
return new ExecutableNotValidNotification();
}
/**
* Returns true if the supplied executable is valid.
* Default implementation: try to execute the given executable and test if output returned errors.
* This can take a long time since it spawns external process.
*
* @param executable Path to executable.
* @return true if process with the supplied executable completed without errors and with exit code 0.
*/
protected boolean isExecutableValid(@NotNull String executable) {
return doCheckExecutable(executable, Collections.<String>emptyList());
}
protected static boolean doCheckExecutable(@NotNull String executable, @NotNull List<String> processParameters) {
try {
GeneralCommandLine commandLine = new GeneralCommandLine();
commandLine.setExePath(executable);
commandLine.addParameters(processParameters);
commandLine.setCharset(CharsetToolkit.getDefaultSystemCharset());
CapturingProcessHandler handler = new CapturingProcessHandler(commandLine);
ProcessOutput result = handler.runProcess(TIMEOUT_MS);
boolean timeout = result.isTimeout();
int exitCode = result.getExitCode();
String stderr = result.getStderr();
if (timeout) {
LOG.warn("Validation of " + executable + " failed with a timeout");
}
if (exitCode != 0) {
LOG.warn("Validation of " + executable + " failed with a non-zero exit code: " + exitCode);
}
if (!stderr.isEmpty()) {
LOG.warn("Validation of " + executable + " failed with a non-empty error output: " + stderr);
}
return !timeout && exitCode == 0 && stderr.isEmpty();
}
catch (Throwable t) {
LOG.warn(t);
return false;
}
}
/**
* Shows a notification about not configured executable with a link to the Settings to fix it.
* Expires the notification if user fixes the path from the opened Settings dialog.
* Makes sure that there is always only one notification about the problem in the stack of notifications.
*/
private void showExecutableNotConfiguredNotification(@NotNull Notification notification) {
if (ApplicationManager.getApplication().isUnitTestMode() || ApplicationManager.getApplication().isHeadlessEnvironment()) {
return;
}
LOG.info("Executable is not valid: " + getCurrentExecutable());
if (myNotificationManager.getNotificationsOfType(notification.getClass(), myProject).length == 0) { // show only once
notification.notify(myProject.isDefault() ? null : myProject);
}
}
@NotNull
protected String prepareDescription(@NotNull String description, boolean appendFixIt) {
StringBuilder result = new StringBuilder();
String executable = getCurrentExecutable();
if (executable.isEmpty()) {
result.append(String.format("<b>%s</b>%s", myNotificationErrorTitle, description));
}
else {
result.append(String.format("<b>%s:</b> <b>%s</b><br/>%s", myNotificationErrorTitle, executable, description));
}
if (appendFixIt) {
result.append(" <a href=''>Fix it.</a>");
}
return result.toString();
}
protected void showSettingsAndExpireIfFixed(@NotNull Notification notification) {
showSettings();
if (validate(getCurrentExecutable()) == null) {
notification.expire();
}
}
protected void showSettings() {
ShowSettingsUtil.getInstance().showSettingsDialog(myProject, getConfigurableDisplayName());
}
/**
* Checks if executable is valid and displays the notification if not.
*
* @return true if executable was valid, false - if not valid (and notification was shown in that case).
* @see #checkExecutableAndShowMessageIfNeeded(java.awt.Component)
*/
public boolean checkExecutableAndNotifyIfNeeded() {
if (myProject.isDisposed()) {
return false;
}
Notification notification = validate(getCurrentExecutable());
return notify(notification);
}
protected boolean notify(@Nullable Notification notification) {
if (notification != null) {
showExecutableNotConfiguredNotification(notification);
return false;
}
return true;
}
/**
* Checks if executable is valid and shows the message if not.
* This method is to be used instead of {@link #checkExecutableAndNotifyIfNeeded()} when Git fails to start from a modal dialog:
* in that case user won't be able to click "Fix it".
*
* @return true if executable was valid, false - if not valid (and a message is shown in that case).
* @see #checkExecutableAndNotifyIfNeeded()
*/
public boolean checkExecutableAndShowMessageIfNeeded(@Nullable Component parentComponent) {
if (myProject.isDisposed()) {
return false;
}
if (!isExecutableValid(getCurrentExecutable())) {
if (Messages.OK == showMessage(parentComponent)) {
ApplicationManager.getApplication().invokeLater(() -> showSettings());
}
return false;
}
return true;
}
private int showMessage(@Nullable Component parentComponent) {
String okText = "Fix it";
String cancelText = CommonBundle.getCancelButtonText();
Icon icon = Messages.getErrorIcon();
String title = myNotificationErrorTitle;
String description = myNotificationErrorDescription;
return parentComponent != null
? Messages.showOkCancelDialog(parentComponent, description, title, okText, cancelText, icon)
: Messages.showOkCancelDialog(myProject, description, title, okText, cancelText, icon);
}
public boolean isExecutableValid() {
return isExecutableValid(getCurrentExecutable());
}
public class ExecutableNotValidNotification extends Notification {
public ExecutableNotValidNotification() {
this(myNotificationErrorDescription);
}
public ExecutableNotValidNotification(@NotNull String description) {
this(prepareDescription(description, true), NotificationType.ERROR);
}
public ExecutableNotValidNotification(@NotNull String preparedDescription, @NotNull NotificationType type) {
super(ourNotificationGroup.getDisplayId(), "", preparedDescription, type, new NotificationListener.Adapter() {
@Override
protected void hyperlinkActivated(@NotNull Notification notification, @NotNull HyperlinkEvent event) {
showSettingsAndExpireIfFixed(notification);
}
});
}
}
}