All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.shell.result.ThrowableResultHandler Maven / Gradle / Ivy

/*
 * Copyright 2015-2022 the original author or authors.
 *
 * 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
 *
 *      https://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 org.springframework.shell.result;

import java.io.PrintWriter;
import java.io.StringWriter;

import org.jline.terminal.Terminal;
import org.jline.utils.AttributedString;
import org.jline.utils.AttributedStringBuilder;
import org.jline.utils.AttributedStyle;

import org.springframework.beans.factory.ObjectProvider;
import org.springframework.shell.ResultHandler;
import org.springframework.shell.command.CommandCatalog;
import org.springframework.shell.context.InteractionMode;
import org.springframework.shell.context.ShellContext;
import org.springframework.shell.jline.InteractiveShellRunner;
import org.springframework.util.StringUtils;

/**
 * A {@link ResultHandler} that prints thrown exceptions messages in red.
 *
 * Stores the last exception reported, so that details can be printed using a
 * dedicated command if in interactive mode. Prints stacktrace if in
 * non-interactive mode as dedicated command could not be used.
 *
 * @author Eric Bottard
 * @author Janne Valkealahti
 */
public class ThrowableResultHandler extends TerminalAwareResultHandler {

	/**
	 * The name of the command that may be used to print details about the last error.
	 */
	public static final String DETAILS_COMMAND_NAME = "stacktrace";

	private Throwable lastError;

	private CommandCatalog commandCatalog;

	private ObjectProvider interactiveRunner;

	private ShellContext shellContext;

	public ThrowableResultHandler(Terminal terminal, CommandCatalog commandCatalog, ShellContext shellContext,
			ObjectProvider interactiveRunner) {
		super(terminal);
		this.commandCatalog = commandCatalog;
		this.shellContext = shellContext;
		this.interactiveRunner = interactiveRunner;
	}

	@Override
	protected void doHandleResult(Throwable result) {
		lastError = result;
		boolean shouldHandle = shouldHandle();

		if (shouldHandle) {
			String errorMsg = StringUtils.hasLength(result.getMessage()) ? result.getMessage() : result.toString();
			terminal.writer().println(new AttributedString(errorMsg,
				AttributedStyle.DEFAULT.foreground(AttributedStyle.RED)).toAnsi());

			String noteMsg;
			if (showShortError()) {
				noteMsg = new AttributedStringBuilder()
					.append("Details of the error have been omitted. You can use the ", AttributedStyle.DEFAULT.foreground(AttributedStyle.RED))
					.append(DETAILS_COMMAND_NAME, AttributedStyle.DEFAULT.foreground(AttributedStyle.RED).bold())
					.append(" command to print the full stacktrace.", AttributedStyle.DEFAULT.foreground(AttributedStyle.RED))
					.toAnsi();
			}
			else {
				StringWriter sw = new StringWriter();
				PrintWriter pw = new PrintWriter(sw);
				result.printStackTrace(pw);
				String stacktraceStr = sw.toString();
				noteMsg = new AttributedString(stacktraceStr,
						AttributedStyle.DEFAULT.foreground(AttributedStyle.RED)).toAnsi();
			}
			terminal.writer().println(noteMsg);
			terminal.writer().flush();
		}
		else {
			if (result instanceof RuntimeException) {
				throw (RuntimeException) result;
			}
			else if (result instanceof Error) {
				throw (Error) result;
			}
			else {
				throw new RuntimeException((Throwable) result);
			}
		}
	}

	/**
	 * Return the last error that was dealt with by this result handler.
	 */
	public Throwable getLastError() {
		return lastError;
	}

	private boolean shouldHandle() {
		return interactiveRunner.getIfAvailable() != null;
	}

	private boolean showShortError() {
		return commandCatalog.getRegistrations().keySet().contains(DETAILS_COMMAND_NAME)
				&& this.shellContext.getInteractionMode() == InteractionMode.INTERACTIVE;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy