commit
a73af52496
|
@ -30,6 +30,7 @@ import java.util.stream.Stream;
|
|||
* A command that can be launched from the layertools jarmode.
|
||||
*
|
||||
* @author Phillip Webb
|
||||
* @author Scott Frederick
|
||||
*/
|
||||
abstract class Command {
|
||||
|
||||
|
@ -192,6 +193,7 @@ abstract class Command {
|
|||
return candidate;
|
||||
}
|
||||
}
|
||||
throw new UnknownOptionException(name);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
@ -285,7 +287,13 @@ abstract class Command {
|
|||
}
|
||||
|
||||
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
|
||||
|
|
|
@ -29,6 +29,7 @@ import org.springframework.boot.loader.jarmode.JarMode;
|
|||
* {@link JarMode} providing {@code "layertools"} support.
|
||||
*
|
||||
* @author Phillip Webb
|
||||
* @author Scott Frederick
|
||||
* @since 2.3.0
|
||||
*/
|
||||
public class LayerToolsJarMode implements JarMode {
|
||||
|
@ -63,22 +64,48 @@ public class LayerToolsJarMode implements JarMode {
|
|||
}
|
||||
|
||||
private void run(String[] args) {
|
||||
run(new ArrayDeque<>(Arrays.asList(args)));
|
||||
run(dequeOf(args));
|
||||
}
|
||||
|
||||
private void run(Deque<String> args) {
|
||||
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) {
|
||||
command.run(args);
|
||||
runCommand(command, args);
|
||||
return;
|
||||
}
|
||||
printError("Unknown command \"" + commandName + "\"");
|
||||
}
|
||||
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) {
|
||||
List<Command> commands = new ArrayList<Command>();
|
||||
List<Command> commands = new ArrayList<>();
|
||||
commands.add(new ListCommand(context));
|
||||
commands.add(new ExtractCommand(context));
|
||||
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.assertThat;
|
||||
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||
|
||||
/**
|
||||
* Tests for {@link Command}.
|
||||
*
|
||||
* @author Phillip Webb
|
||||
* @author Scott Frederick
|
||||
*/
|
||||
class CommandTests {
|
||||
|
||||
|
@ -77,6 +79,20 @@ class CommandTests {
|
|||
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
|
||||
void findWhenNameMatchesReturnsCommand() {
|
||||
TestCommand test1 = new TestCommand("test1");
|
||||
|
|
|
@ -39,6 +39,7 @@ import static org.mockito.Mockito.mock;
|
|||
* Tests for {@link LayerToolsJarMode}.
|
||||
*
|
||||
* @author Phillip Webb
|
||||
* @author Scott Frederick
|
||||
*/
|
||||
class LayerToolsJarModeTests {
|
||||
|
||||
|
@ -79,6 +80,24 @@ class LayerToolsJarModeTests {
|
|||
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 {
|
||||
File file = new File(this.temp, name);
|
||||
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