commit
a73af52496
|
@ -30,6 +30,7 @@ import java.util.stream.Stream;
|
||||||
* A command that can be launched from the layertools jarmode.
|
* A command that can be launched from the layertools jarmode.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
|
* @author Scott Frederick
|
||||||
*/
|
*/
|
||||||
abstract class Command {
|
abstract class Command {
|
||||||
|
|
||||||
|
@ -192,6 +193,7 @@ abstract class Command {
|
||||||
return candidate;
|
return candidate;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
throw new UnknownOptionException(name);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -285,7 +287,13 @@ abstract class Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
private String claimArg(Deque<String> args) {
|
private String claimArg(Deque<String> args) {
|
||||||
return (this.valueDescription != null) ? args.removeFirst() : null;
|
if (this.valueDescription != null) {
|
||||||
|
if (args.isEmpty()) {
|
||||||
|
throw new MissingValueException(this.name);
|
||||||
|
}
|
||||||
|
return args.removeFirst();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -29,6 +29,7 @@ import org.springframework.boot.loader.jarmode.JarMode;
|
||||||
* {@link JarMode} providing {@code "layertools"} support.
|
* {@link JarMode} providing {@code "layertools"} support.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
|
* @author Scott Frederick
|
||||||
* @since 2.3.0
|
* @since 2.3.0
|
||||||
*/
|
*/
|
||||||
public class LayerToolsJarMode implements JarMode {
|
public class LayerToolsJarMode implements JarMode {
|
||||||
|
@ -63,22 +64,48 @@ public class LayerToolsJarMode implements JarMode {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void run(String[] args) {
|
private void run(String[] args) {
|
||||||
run(new ArrayDeque<>(Arrays.asList(args)));
|
run(dequeOf(args));
|
||||||
}
|
}
|
||||||
|
|
||||||
private void run(Deque<String> args) {
|
private void run(Deque<String> args) {
|
||||||
if (!args.isEmpty()) {
|
if (!args.isEmpty()) {
|
||||||
Command command = Command.find(this.commands, args.removeFirst());
|
String commandName = args.removeFirst();
|
||||||
|
Command command = Command.find(this.commands, commandName);
|
||||||
if (command != null) {
|
if (command != null) {
|
||||||
command.run(args);
|
runCommand(command, args);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
printError("Unknown command \"" + commandName + "\"");
|
||||||
}
|
}
|
||||||
this.help.run(args);
|
this.help.run(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void runCommand(Command command, Deque<String> args) {
|
||||||
|
try {
|
||||||
|
command.run(args);
|
||||||
|
}
|
||||||
|
catch (UnknownOptionException ex) {
|
||||||
|
printError("Unknown option \"" + ex.getMessage() + "\" for the " + command.getName() + " command");
|
||||||
|
this.help.run(dequeOf(command.getName()));
|
||||||
|
}
|
||||||
|
catch (MissingValueException ex) {
|
||||||
|
printError("Option \"" + ex.getMessage() + "\" for the " + command.getName()
|
||||||
|
+ " command requires a value");
|
||||||
|
this.help.run(dequeOf(command.getName()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void printError(String errorMessage) {
|
||||||
|
System.out.println("Error: " + errorMessage);
|
||||||
|
System.out.println();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Deque<String> dequeOf(String... args) {
|
||||||
|
return new ArrayDeque<>(Arrays.asList(args));
|
||||||
|
}
|
||||||
|
|
||||||
static List<Command> getCommands(Context context) {
|
static List<Command> getCommands(Context context) {
|
||||||
List<Command> commands = new ArrayList<Command>();
|
List<Command> commands = new ArrayList<>();
|
||||||
commands.add(new ListCommand(context));
|
commands.add(new ListCommand(context));
|
||||||
commands.add(new ExtractCommand(context));
|
commands.add(new ExtractCommand(context));
|
||||||
return Collections.unmodifiableList(commands);
|
return Collections.unmodifiableList(commands);
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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.boot.jarmode.layertools;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exception thrown when a required value is not provided for an option.
|
||||||
|
*
|
||||||
|
* @author Scott Frederick
|
||||||
|
*/
|
||||||
|
class MissingValueException extends RuntimeException {
|
||||||
|
|
||||||
|
private final String optionName;
|
||||||
|
|
||||||
|
MissingValueException(String optionName) {
|
||||||
|
this.optionName = optionName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getMessage() {
|
||||||
|
return "--" + this.optionName;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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.boot.jarmode.layertools;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exception thrown when an unrecognized option is encountered.
|
||||||
|
*
|
||||||
|
* @author Scott Frederick
|
||||||
|
*/
|
||||||
|
class UnknownOptionException extends RuntimeException {
|
||||||
|
|
||||||
|
private final String optionName;
|
||||||
|
|
||||||
|
UnknownOptionException(String optionName) {
|
||||||
|
this.optionName = optionName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getMessage() {
|
||||||
|
return "--" + this.optionName;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -30,11 +30,13 @@ import org.springframework.boot.jarmode.layertools.Command.Parameters;
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.as;
|
import static org.assertj.core.api.Assertions.as;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests for {@link Command}.
|
* Tests for {@link Command}.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
|
* @author Scott Frederick
|
||||||
*/
|
*/
|
||||||
class CommandTests {
|
class CommandTests {
|
||||||
|
|
||||||
|
@ -77,6 +79,20 @@ class CommandTests {
|
||||||
assertThat(command.getRunParameters()).containsExactly("test2", "test3");
|
assertThat(command.getRunParameters()).containsExactly("test2", "test3");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void runWithUnknownOptionThrowsException() {
|
||||||
|
TestCommand command = new TestCommand("test", VERBOSE_FLAG, LOG_LEVEL_OPTION);
|
||||||
|
assertThatExceptionOfType(UnknownOptionException.class).isThrownBy(() -> run(command, "--invalid"))
|
||||||
|
.withMessage("--invalid");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void runWithOptionMissingRequiredValueThrowsException() {
|
||||||
|
TestCommand command = new TestCommand("test", VERBOSE_FLAG, LOG_LEVEL_OPTION);
|
||||||
|
assertThatExceptionOfType(MissingValueException.class)
|
||||||
|
.isThrownBy(() -> run(command, "--verbose", "--log-level")).withMessage("--log-level");
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void findWhenNameMatchesReturnsCommand() {
|
void findWhenNameMatchesReturnsCommand() {
|
||||||
TestCommand test1 = new TestCommand("test1");
|
TestCommand test1 = new TestCommand("test1");
|
||||||
|
|
|
@ -39,6 +39,7 @@ import static org.mockito.Mockito.mock;
|
||||||
* Tests for {@link LayerToolsJarMode}.
|
* Tests for {@link LayerToolsJarMode}.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
|
* @author Scott Frederick
|
||||||
*/
|
*/
|
||||||
class LayerToolsJarModeTests {
|
class LayerToolsJarModeTests {
|
||||||
|
|
||||||
|
@ -79,6 +80,24 @@ class LayerToolsJarModeTests {
|
||||||
assertThat(this.out).hasSameContentAsResource("list-output.txt");
|
assertThat(this.out).hasSameContentAsResource("list-output.txt");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void mainWithUnknownCommandShowsErrorAndHelp() {
|
||||||
|
new LayerToolsJarMode().run("layertools", new String[] { "invalid" });
|
||||||
|
assertThat(this.out).hasSameContentAsResource("error-command-unknown-output.txt");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void mainWithUnknownOptionShowsErrorAndCommandHelp() {
|
||||||
|
new LayerToolsJarMode().run("layertools", new String[] { "extract", "--invalid" });
|
||||||
|
assertThat(this.out).hasSameContentAsResource("error-option-unknown-output.txt");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void mainWithOptionMissingRequiredValueShowsErrorAndCommandHelp() {
|
||||||
|
new LayerToolsJarMode().run("layertools", new String[] { "extract", "--destination" });
|
||||||
|
assertThat(this.out).hasSameContentAsResource("error-option-missing-value-output.txt");
|
||||||
|
}
|
||||||
|
|
||||||
private File createJarFile(String name) throws IOException {
|
private File createJarFile(String name) throws IOException {
|
||||||
File file = new File(this.temp, name);
|
File file = new File(this.temp, name);
|
||||||
try (ZipOutputStream jarOutputStream = new ZipOutputStream(new FileOutputStream(file))) {
|
try (ZipOutputStream jarOutputStream = new ZipOutputStream(new FileOutputStream(file))) {
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
Error: Unknown command "invalid"
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
java -Djarmode=layertools -jar test.jar
|
||||||
|
|
||||||
|
Available commands:
|
||||||
|
list List layers from the jar that can be extracted
|
||||||
|
extract Extracts layers from the jar for image creation
|
||||||
|
help Help about any command
|
|
@ -0,0 +1,9 @@
|
||||||
|
Error: Option "--destination" for the extract command requires a value
|
||||||
|
|
||||||
|
Extracts layers from the jar for image creation
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
java -Djarmode=layertools -jar test.jar extract [options] [<layer>...]
|
||||||
|
|
||||||
|
Options:
|
||||||
|
--destination string The destination to extract files to
|
|
@ -0,0 +1,9 @@
|
||||||
|
Error: Unknown option "--invalid" for the extract command
|
||||||
|
|
||||||
|
Extracts layers from the jar for image creation
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
java -Djarmode=layertools -jar test.jar extract [options] [<layer>...]
|
||||||
|
|
||||||
|
Options:
|
||||||
|
--destination string The destination to extract files to
|
Loading…
Reference in New Issue