执行单个命令并退出 Spring Shell 2
Executing a single command and exiting from Spring Shell 2
我不久前偶然发现了 this question,解释了如何让 Spring Shell 应用程序在使用单个命令从命令行调用后退出。然而,在 2.0.0 中使用 Spring Boot 对此进行测试,似乎不再是使用命令参数调用 JAR 将执行该命令然后退出的情况。 shell 只是正常启动而不执行提供的命令。仍然可以这样做吗?如果不是,是否可以将参数从 JAR 执行传递给 Spring Shell 然后在执行后触发退出?
例如,假设我有一个命令 import
,它有几个选项。 shell 中的 运行 可能是这样的:
$ java -jar my-app.jar
> import -f /path/to/file.txt --overwrite
Successfully imported 'file.txt'
> exit
但是如果能够简单地执行和退出就更好了,为了构建一个可以利用此功能的脚本:
$ java -jar my-app.jar import -f /path/to/file.txt --overwrite
Successfully imported 'file.txt'
我找到了一个不错的小解决方法。我没有创建一个模仿 v1 行为的 ApplicationRunner
(这很棘手,因为 JLineInputProvider
是私有的 class),我创建了一个可选加载的,基于 active Spring 轮廓。我使用 JCommander 来定义 CLI 参数,允许我对交互式 shell 和一次性执行使用相同的命令。 运行 没有参数的 Spring 引导 JAR 触发交互式 shell。 运行 它与参数一起触发一次性执行。
@Parameters
public class ImportParameters {
@Parameter(names = { "-f", "--file" }, required = true, description = "Data file")
private File file;
@Parameter(names = { "-t", "--type" }, required = true, description = "Data type")
private DataType dataType;
@Parameter(names = { "-o", "--overwrite" }, description = "Flag to overwrite file if it exists")
private Boolean overwrite = false;
/* getters and setters */
}
public class ImportCommandExecutor {
public void run(ImportParameters params) throws Exception {
// import logic goes here
}
}
/* Handles interactive shell command execution */
@ShellComponent
public class JLineInputExecutor {
// All command executors are injected here
@Autowired private ImportCommandExecutor importExecutor;
...
@ShellMethod(key = "import", value = "Imports the a file of a specified type.")
public String importCommand(@ShellOption(optOut = true) ImportParameters params) throws Exception {
importCommandExecutor.run(params);
}
...
}
/* Handles one-off command execution */
public class JCommanderInputExecutor implements ApplicationRunner {
// All command executors are injected here
@Autowired private ImportCommandExecutor importExecutor;
...
@Override
public void run(ApplicationArguments args) throws Exception {
// Create all of the JCommander argument handler objects
BaseParameters baseParameters = new BaseParameters();
ImportParameters importParameters = new ImportParameters();
...
JCommander jc = newBuilder().
.acceptUnknownOptions(true)
.addObject(baseParameters)
.addCommand("import", importParameters)
...
.build();
jc.parse(args);
String mainCommand = jc.getParsedCommand();
if ("import".equals(mainCommand)){
importExecutor.run(importParameters);
} else if (...) {
...
}
}
}
@Configuration
@Profile({"CLI"})
public class CommandLineInterfaceConfiguration {
// All of my command executors are defined as beans here, as well as other required configurations for both modes of execution
@Bean
public ImportCommandExecutor importExecutor (){
return new ImportCommandExecutor();
}
...
}
@Configuration
@Profile({"SINGLE_COMMAND"})
public class SingleCommandConfiguration {
@Bean
public JCommanderInputExecutor commandLineInputExecutor(){
return new JCommanderInputExecutor();
}
}
@SpringBootApplication
public class Application {
public static void main(String[] args) throws IOException {
String[] profiles = getActiveProfiles(args);
SpringApplicationBuilder builder = new SpringApplicationBuilder(Application.class);
builder.bannerMode((Mode.LOG));
builder.web(false);
builder.profiles(profiles);
System.out.println(String.format("Command line arguments: %s Profiles: %s",
Arrays.asList(args), Arrays.asList(profiles)));
builder.run(args);
}
private static String[] getActiveProfiles(String[] args){
return Arrays.asList(args).contains("-X") ? new String[]{"CLI", "SINGLE_COMMAND"} : new String[]{"CLI"};
}
}
所以现在我可以通过 运行 我的可执行 JAR 来触发交互式客户端:
java -jar app.jar
> import -f /path/to/file.txt -t GENE -o
> quit()
或者,如果我在命令行上传递“-X”参数,应用程序将执行然后退出:
java -jar app.jar -X import -f /path/to/file.txt -t GENE -o
补充一下,我找到了另一种方法,在交互模式下不提供 运行 选项,但使用上面的配置文件当然可以交换配置。请注意我正在使用 lombok 和 jool(以防万一有人复制粘贴并遇到有趣的问题!)
条目
@SpringBootApplication
public class Righter {
public static void main(String[] args) {
SpringApplication.run(Righter.class, args);
}
@Bean
public ApplicationRunner shellRunner(Shell shell) {
return new NonInteractiveShellRunner(shell);
}
申请运行纳尔:
@Order(0)
public class NonInteractiveShellRunner implements ApplicationRunner{
private final Shell shell;
public NonInteractiveShellRunner(Shell shell) {
this.shell = shell;
}
@Override
public void run(ApplicationArguments args) throws Exception {
shell.run(new CommandInputProvider(args.getSourceArgs()));
}
public static class PredefinedInputProvider implements InputProvider{
private final Input input;
private boolean commandExecuted = false;
public PredefinedInputProvider(String[] args) {
this.input = new PredefinedInput(args);
}
@Override
public Input readInput() {
if (!commandExecuted){
commandExecuted=true;
return input;
}
return new PredefinedInput(new String[]{"exit"});
}
@AllArgsConstructor
private static class PredefinedInput implements Input{
private final String[] args;
@Override
public String rawText() {
return Seq.of(args).toString(" ");
}
@Override
public List<String> words(){
return Arrays.asList(args);
}
}
}
}
除了 Alex 的回答,这里是我制作的 NonInteractiveApplicationRunner
的更简单版本。
@Component
@Order(InteractiveShellApplicationRunner.PRECEDENCE - 100)
class NonInteractiveApplicationRunner implements ApplicationRunner {
private final Shell shell;
private final ConfigurableEnvironment environment;
public NonInteractiveApplicationRunner(Shell shell, ConfigurableEnvironment environment) {
this.shell = shell;
this.environment = environment;
}
@Override
public void run(ApplicationArguments args) {
if (args.getSourceArgs().length > 0) {
InteractiveShellApplicationRunner.disable(environment);
var input = String.join(" ", args.getSourceArgs());
shell.evaluate(() -> input);
shell.evaluate(() -> "exit");
}
}
}
使用@Component
,我们不需要添加bean方法。此外,与 shell.run(...)
.
相比,使用 shell.evaluate()
方法看起来简单得多
运行 使用@my-script,像这样:
java -jar my-app.jar @my-script
我的脚本是包含您的命令的文件:
import -f /path/to/file.txt --overwrite
一种添加单个命令运行模式的方法,不排除交互模式和脚本模式(在spring-shell-starter::2.0.0.RELEASE上测试).
类比ScriptShellApplication运行ner创建运行ner.
// Runs before ScriptShellApplicationRunner and InteractiveShellApplicationRunner
@Order(InteractiveShellApplicationRunner.PRECEDENCE - 200)
public class SingleCommandApplicationRunner implements ApplicationRunner {
private final Parser parser;
private final Shell shell;
private final ConfigurableEnvironment environment;
private final Set<String> allCommandNames;
public SingleCommandApplicationRunner(
Parser parser,
Shell shell,
ConfigurableEnvironment environment,
Set<CustomCommand> customCommands
) {
this.parser = parser;
this.shell = shell;
this.environment = environment;
this.allCommandNames = buildAllCommandNames(customCommands);
}
private Set<String> buildAllCommandNames(Collection<CustomCommand> customCommands) {
final Set<String> result = new HashSet<>();
customCommands.stream().map(CustomCommand::keys).flatMap(Collection::stream).forEach(result::add);
// default spring shell commands
result.addAll(asList("clear", "exit", "quit", "help", "script", "stacktrace"));
return result;
}
@Override
public void run(ApplicationArguments args) throws Exception {
final boolean singleCommand = haveCommand(args.getSourceArgs());
if (singleCommand) {
InteractiveShellApplicationRunner.disable(environment);
final String fullArgs = join(" ", args.getSourceArgs());
try (Reader reader = new StringReader(fullArgs);
FileInputProvider inputProvider = new FileInputProvider(reader, parser)) {
shell.run(inputProvider);
}
}
}
private boolean haveCommand(String... args) {
for (String arg : args) {
if (allCommandNames.contains(arg)) {
return true;
}
}
return false;
}
}
将 运行ner 注册为 bean。
@Configuration
class ContextConfiguration {
@Autowired
private Shell shell;
@Bean
SingleCommandApplicationRunner singleCommandApplicationRunner(
Parser parser,
ConfigurableEnvironment environment,
Set<CustomCommand> customCommands
) {
return new SingleCommandApplicationRunner(parser, shell, environment, customCommands);
}
}
以便 运行ner 仅在发送命令时启动,我们创建一个接口。
public interface CustomCommand {
Collection<String> keys();
}
在每个命令中实现 CustomCommand 接口。
@ShellComponent
@RequiredArgsConstructor
class MyCommand implements CustomCommand {
private static final String KEY = "my-command";
@Override
public Collection<String> keys() {
return singletonList(KEY);
}
@ShellMethod(key = KEY, value = "My custom command.")
public AttributedString version() {
return "Hello, single command mode!";
}
}
完成!
运行 交互模式:
java -jar myApp.jar
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// shell:>my-command
// Hello, single command mode!
运行 来自文件 script.txt 的脚本(包含文本“my-command”):
java -jar myApp.jar @script.txt
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// Hello, single command mode!
运行 单命令模式:
java -jar myApp.jar my-command
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// Hello, single command mode!
在 linux 中也是这样工作的:
echo "import -f /path/to/file.txt --overwrite" | java -jar my-app.jar
尽管它以失败告终;它仅在成功执行命令后发生。
我不久前偶然发现了 this question,解释了如何让 Spring Shell 应用程序在使用单个命令从命令行调用后退出。然而,在 2.0.0 中使用 Spring Boot 对此进行测试,似乎不再是使用命令参数调用 JAR 将执行该命令然后退出的情况。 shell 只是正常启动而不执行提供的命令。仍然可以这样做吗?如果不是,是否可以将参数从 JAR 执行传递给 Spring Shell 然后在执行后触发退出?
例如,假设我有一个命令 import
,它有几个选项。 shell 中的 运行 可能是这样的:
$ java -jar my-app.jar
> import -f /path/to/file.txt --overwrite
Successfully imported 'file.txt'
> exit
但是如果能够简单地执行和退出就更好了,为了构建一个可以利用此功能的脚本:
$ java -jar my-app.jar import -f /path/to/file.txt --overwrite
Successfully imported 'file.txt'
我找到了一个不错的小解决方法。我没有创建一个模仿 v1 行为的 ApplicationRunner
(这很棘手,因为 JLineInputProvider
是私有的 class),我创建了一个可选加载的,基于 active Spring 轮廓。我使用 JCommander 来定义 CLI 参数,允许我对交互式 shell 和一次性执行使用相同的命令。 运行 没有参数的 Spring 引导 JAR 触发交互式 shell。 运行 它与参数一起触发一次性执行。
@Parameters
public class ImportParameters {
@Parameter(names = { "-f", "--file" }, required = true, description = "Data file")
private File file;
@Parameter(names = { "-t", "--type" }, required = true, description = "Data type")
private DataType dataType;
@Parameter(names = { "-o", "--overwrite" }, description = "Flag to overwrite file if it exists")
private Boolean overwrite = false;
/* getters and setters */
}
public class ImportCommandExecutor {
public void run(ImportParameters params) throws Exception {
// import logic goes here
}
}
/* Handles interactive shell command execution */
@ShellComponent
public class JLineInputExecutor {
// All command executors are injected here
@Autowired private ImportCommandExecutor importExecutor;
...
@ShellMethod(key = "import", value = "Imports the a file of a specified type.")
public String importCommand(@ShellOption(optOut = true) ImportParameters params) throws Exception {
importCommandExecutor.run(params);
}
...
}
/* Handles one-off command execution */
public class JCommanderInputExecutor implements ApplicationRunner {
// All command executors are injected here
@Autowired private ImportCommandExecutor importExecutor;
...
@Override
public void run(ApplicationArguments args) throws Exception {
// Create all of the JCommander argument handler objects
BaseParameters baseParameters = new BaseParameters();
ImportParameters importParameters = new ImportParameters();
...
JCommander jc = newBuilder().
.acceptUnknownOptions(true)
.addObject(baseParameters)
.addCommand("import", importParameters)
...
.build();
jc.parse(args);
String mainCommand = jc.getParsedCommand();
if ("import".equals(mainCommand)){
importExecutor.run(importParameters);
} else if (...) {
...
}
}
}
@Configuration
@Profile({"CLI"})
public class CommandLineInterfaceConfiguration {
// All of my command executors are defined as beans here, as well as other required configurations for both modes of execution
@Bean
public ImportCommandExecutor importExecutor (){
return new ImportCommandExecutor();
}
...
}
@Configuration
@Profile({"SINGLE_COMMAND"})
public class SingleCommandConfiguration {
@Bean
public JCommanderInputExecutor commandLineInputExecutor(){
return new JCommanderInputExecutor();
}
}
@SpringBootApplication
public class Application {
public static void main(String[] args) throws IOException {
String[] profiles = getActiveProfiles(args);
SpringApplicationBuilder builder = new SpringApplicationBuilder(Application.class);
builder.bannerMode((Mode.LOG));
builder.web(false);
builder.profiles(profiles);
System.out.println(String.format("Command line arguments: %s Profiles: %s",
Arrays.asList(args), Arrays.asList(profiles)));
builder.run(args);
}
private static String[] getActiveProfiles(String[] args){
return Arrays.asList(args).contains("-X") ? new String[]{"CLI", "SINGLE_COMMAND"} : new String[]{"CLI"};
}
}
所以现在我可以通过 运行 我的可执行 JAR 来触发交互式客户端:
java -jar app.jar
> import -f /path/to/file.txt -t GENE -o
> quit()
或者,如果我在命令行上传递“-X”参数,应用程序将执行然后退出:
java -jar app.jar -X import -f /path/to/file.txt -t GENE -o
补充一下,我找到了另一种方法,在交互模式下不提供 运行 选项,但使用上面的配置文件当然可以交换配置。请注意我正在使用 lombok 和 jool(以防万一有人复制粘贴并遇到有趣的问题!)
条目
@SpringBootApplication
public class Righter {
public static void main(String[] args) {
SpringApplication.run(Righter.class, args);
}
@Bean
public ApplicationRunner shellRunner(Shell shell) {
return new NonInteractiveShellRunner(shell);
}
申请运行纳尔:
@Order(0)
public class NonInteractiveShellRunner implements ApplicationRunner{
private final Shell shell;
public NonInteractiveShellRunner(Shell shell) {
this.shell = shell;
}
@Override
public void run(ApplicationArguments args) throws Exception {
shell.run(new CommandInputProvider(args.getSourceArgs()));
}
public static class PredefinedInputProvider implements InputProvider{
private final Input input;
private boolean commandExecuted = false;
public PredefinedInputProvider(String[] args) {
this.input = new PredefinedInput(args);
}
@Override
public Input readInput() {
if (!commandExecuted){
commandExecuted=true;
return input;
}
return new PredefinedInput(new String[]{"exit"});
}
@AllArgsConstructor
private static class PredefinedInput implements Input{
private final String[] args;
@Override
public String rawText() {
return Seq.of(args).toString(" ");
}
@Override
public List<String> words(){
return Arrays.asList(args);
}
}
}
}
除了 Alex 的回答,这里是我制作的 NonInteractiveApplicationRunner
的更简单版本。
@Component
@Order(InteractiveShellApplicationRunner.PRECEDENCE - 100)
class NonInteractiveApplicationRunner implements ApplicationRunner {
private final Shell shell;
private final ConfigurableEnvironment environment;
public NonInteractiveApplicationRunner(Shell shell, ConfigurableEnvironment environment) {
this.shell = shell;
this.environment = environment;
}
@Override
public void run(ApplicationArguments args) {
if (args.getSourceArgs().length > 0) {
InteractiveShellApplicationRunner.disable(environment);
var input = String.join(" ", args.getSourceArgs());
shell.evaluate(() -> input);
shell.evaluate(() -> "exit");
}
}
}
使用@Component
,我们不需要添加bean方法。此外,与 shell.run(...)
.
shell.evaluate()
方法看起来简单得多
运行 使用@my-script,像这样:
java -jar my-app.jar @my-script
我的脚本是包含您的命令的文件:
import -f /path/to/file.txt --overwrite
一种添加单个命令运行模式的方法,不排除交互模式和脚本模式(在spring-shell-starter::2.0.0.RELEASE上测试).
类比ScriptShellApplication运行ner创建运行ner.
// Runs before ScriptShellApplicationRunner and InteractiveShellApplicationRunner
@Order(InteractiveShellApplicationRunner.PRECEDENCE - 200)
public class SingleCommandApplicationRunner implements ApplicationRunner {
private final Parser parser;
private final Shell shell;
private final ConfigurableEnvironment environment;
private final Set<String> allCommandNames;
public SingleCommandApplicationRunner(
Parser parser,
Shell shell,
ConfigurableEnvironment environment,
Set<CustomCommand> customCommands
) {
this.parser = parser;
this.shell = shell;
this.environment = environment;
this.allCommandNames = buildAllCommandNames(customCommands);
}
private Set<String> buildAllCommandNames(Collection<CustomCommand> customCommands) {
final Set<String> result = new HashSet<>();
customCommands.stream().map(CustomCommand::keys).flatMap(Collection::stream).forEach(result::add);
// default spring shell commands
result.addAll(asList("clear", "exit", "quit", "help", "script", "stacktrace"));
return result;
}
@Override
public void run(ApplicationArguments args) throws Exception {
final boolean singleCommand = haveCommand(args.getSourceArgs());
if (singleCommand) {
InteractiveShellApplicationRunner.disable(environment);
final String fullArgs = join(" ", args.getSourceArgs());
try (Reader reader = new StringReader(fullArgs);
FileInputProvider inputProvider = new FileInputProvider(reader, parser)) {
shell.run(inputProvider);
}
}
}
private boolean haveCommand(String... args) {
for (String arg : args) {
if (allCommandNames.contains(arg)) {
return true;
}
}
return false;
}
}
将 运行ner 注册为 bean。
@Configuration
class ContextConfiguration {
@Autowired
private Shell shell;
@Bean
SingleCommandApplicationRunner singleCommandApplicationRunner(
Parser parser,
ConfigurableEnvironment environment,
Set<CustomCommand> customCommands
) {
return new SingleCommandApplicationRunner(parser, shell, environment, customCommands);
}
}
以便 运行ner 仅在发送命令时启动,我们创建一个接口。
public interface CustomCommand {
Collection<String> keys();
}
在每个命令中实现 CustomCommand 接口。
@ShellComponent
@RequiredArgsConstructor
class MyCommand implements CustomCommand {
private static final String KEY = "my-command";
@Override
public Collection<String> keys() {
return singletonList(KEY);
}
@ShellMethod(key = KEY, value = "My custom command.")
public AttributedString version() {
return "Hello, single command mode!";
}
}
完成!
运行 交互模式:
java -jar myApp.jar
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// shell:>my-command
// Hello, single command mode!
运行 来自文件 script.txt 的脚本(包含文本“my-command”):
java -jar myApp.jar @script.txt
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// Hello, single command mode!
运行 单命令模式:
java -jar myApp.jar my-command
// 2021-01-14 19:28:16.911 INFO 67313 --- [main] com.nao4j.example.Application: Starting Application v1.0.0 using Java 1.8.0_275 on Apple-MacBook-Pro-15.local with PID 67313 (/Users/nao4j/example/target/myApp.jar started by nao4j in /Users/nao4j/example/target)
// 2021-01-14 19:28:16.916 INFO 67313 --- [main] com.nao4j.example.Application: No active profile set, falling back to default profiles: default
// 2021-01-14 19:28:18.227 INFO 67313 --- [main] com.nao4j.example.Application: Started Application in 2.179 seconds (JVM running for 2.796)
// Hello, single command mode!
在 linux 中也是这样工作的:
echo "import -f /path/to/file.txt --overwrite" | java -jar my-app.jar
尽管它以失败告终;它仅在成功执行命令后发生。