Picocli+Spring Boot でコマンドラインアプリケーションを作成してみる
概要
記事一覧はこちらです。
Twitter を見ていたところ picocli というライブラリの 4.0 GA release のツイートを見かけました。picocli のことを知らなかったので調べてみたところ、
- Java で command line application を作成するための framework。考えられるものはほとんど実装されているのではないだろうか、というぐらい機能が充実している。マニュアルも分かりやすい。
- picocli-spring-boot-starter が提供されている。
- bash か zsh 限定だが、コマンドラインから実行する時に TAB で自動補完が効くコマンドを作成できる。
- GraalVM に対応しているらしい。GraalVM を全然理解していないので自分ではどう対応されているのか全く分かりませんでしたが。。。
調べたことをメモして残し、サンプルを作成してみます。作成したサンプルは以下の URL の場所に置いてあります。
https://github.com/ksby/ksbysample-boot-miscellaneous/tree/master/picocli-boot-cmdapp
参照したサイト・書籍
picocli - a mighty tiny command line interface
https://picocli.info/Quick Guide
https://picocli.info/quick-guide.htmlremkop/picocli
https://github.com/remkop/picoclipicocli/picocli-spring-boot-starter/
https://github.com/remkop/picocli/tree/master/picocli-spring-boot-starterAutocomplete for Java Command Line Applications
https://picocli.info/autocomplete.htmlCreate a Java Command Line Program with Picocli
https://www.baeldung.com/java-picocli-create-command-line-programSpring Boot Exit Codes
https://www.baeldung.com/spring-boot-exit-codesIncluding subprojects using a wildcard in a Gradle settings file
https://stackoverflow.com/questions/2297032/including-subprojects-using-a-wildcard-in-a-gradle-settings-file
目次
- Spring Boot ベースのコマンドラインアプリケーションのサンプルを作成する(Subcommand なし)
- Spring Boot ベースのコマンドラインアプリケーションのサンプルを作成する(Subcommand あり)
- --version(-V)オプション指定時に build.gradle に記述した build.version を表示する
- TAB キー押下時に subcommand, option の候補の表示や自動補完が行われるようにする
手順
Spring Boot ベースのコマンドラインアプリケーションのサンプルを作成する(Subcommand なし)
Subcommand なしと Subcommand ありの2つのサンプルを Gradle Multi-project の中に作成します。
D:\project-springboot\ksbysample-boot-miscellaneous
の下に picocli-boot-cmdapp ディレクトリを作成する。- 別のプロジェクトから Gradle Wrapper のファイルをコピーする(コピーしたのは 5.4.1)。
- Gradle を最新バージョン(5.5.1)にする。
gradlew init
を実行する。- settings.gradle を以下の内容に変更する(Including subprojects using a wildcard in a Gradle settings file 参照)。これで build.gradle があるサブプロジェクトは include 分を記述しなくても自動的に Multi-project に認識されるようになる。
rootProject.name = 'picocli-boot-cmdapp' rootDir.eachFileRecurse { f -> if ( f.name == "build.gradle" ) { String relativePath = f.parentFile.absolutePath - rootDir.absolutePath String projectName = relativePath.replaceAll("[\\\\\\/]", ":") include projectName } }
Multi-project のベースが出来ました。次に Spring Boot ベースのコマンドラインアプリケーション(Subcommand なし)の Project を作成します。以下の仕様のコマンドを作成します。
filetools --create <ファイル> <ファイル> ...
(--create は -c も可) で指定されたファイル名の空ファイルを作成する。filetools --delete <ファイル> <ファイル> ...
(--delete は -d も可) で指定されたファイルを削除する。--create
と--delete
のオプションはいずれか一方が必須。どちらも指定しない、あるいはどちらも指定した場合にはエラーになる。
まず D:\project-springboot\ksbysample-boot-miscellaneous\picocli-boot-cmdapp
の下に Spring Initializr で nosubcmd-cmdapp プロジェクトを作成した後、build.gradle を以下のように変更します。
buildscript { ext { group "ksby.ksbysample-boot-miscellaneous.picocli-boot-cmdapp" version "1.0.0-RELEASE" } repositories { mavenCentral() maven { url "https://repo.spring.io/release/" } maven { url "https://plugins.gradle.org/m2/" } } } plugins { id 'org.springframework.boot' version '2.1.6.RELEASE' id "io.spring.dependency-management" version "1.0.8.RELEASE" id 'java' id 'idea' } sourceCompatibility = JavaVersion.VERSION_11 targetCompatibility = JavaVersion.VERSION_11 idea { module { inheritOutputDirs = false outputDir = file("$buildDir/classes/main/") } } repositories { mavenCentral() } dependencyManagement { imports { mavenBom(org.springframework.boot.gradle.plugin.SpringBootPlugin.BOM_COORDINATES) } } dependencies { def picocliVersion = "4.0.0" implementation("org.springframework.boot:spring-boot-starter") testImplementation("org.springframework.boot:spring-boot-starter-test") // picocli implementation("info.picocli:picocli-spring-boot-starter:${picocliVersion}") }
- dependencies block に Picocli の Spring Boot Starter である picocli-spring-boot-starter を追加します。
src/main/java/ksbysample/cmdapp/nosubcmd の下に FileToolsCommand.java を新規作成して、以下の内容を記述します。
package ksbysample.cmdapp.nosubcmd; import org.springframework.stereotype.Component; import picocli.CommandLine.*; import java.io.File; import java.io.IOException; import java.nio.file.FileAlreadyExistsException; import java.nio.file.FileSystemException; import java.nio.file.Files; import java.nio.file.Paths; import java.util.Arrays; import java.util.concurrent.Callable; @Component @Command(name = "filetools", mixinStandardHelpOptions = true, version = "1.0.0", description = "create/delete file(s) command") public class FileToolsCommand implements Callable<Integer>, IExitCodeExceptionMapper { // --create オプションと --delete オプションはいずれか一方しか指定できないようにする @ArgGroup(exclusive = true, multiplicity = "1") private Exclusive exclusive; static class Exclusive { @Option(names = {"-c", "--create"}, description = "create file(s)") private boolean isCreate; @Option(names = {"-d", "--delete"}, description = "delete file(s)") private boolean isDelete; } @Parameters(paramLabel = "ファイル", description = "作成あるいは削除するファイル") private File[] files; @Override public Integer call() { Arrays.asList(this.files).forEach(f -> { try { if (exclusive.isCreate) { Files.createFile(Paths.get(f.getName())); System.out.println(f.getName() + " is created."); } else if (exclusive.isDelete) { Files.deleteIfExists(Paths.get(f.getName())); System.out.println(f.getName() + " is deleted."); } } catch (IOException e) { throw new RuntimeException(e); } }); return ExitCode.OK; } @Override public int getExitCode(Throwable exception) { Throwable cause = exception.getCause(); if (cause instanceof FileAlreadyExistsException) { // 既に存在するファイルを作成しようとしている return 12; } else if (cause instanceof FileSystemException) { // 削除しようとしたファイルが別のプロセスでオープンされている等 return 13; } return 11; } }
src/main/java/ksbysample/cmdapp/nosubcmd/Application.java を以下のように変更します。
package ksbysample.cmdapp.nosubcmd; import org.springframework.boot.CommandLineRunner; import org.springframework.boot.ExitCodeGenerator; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import picocli.CommandLine; import picocli.CommandLine.IFactory; @SpringBootApplication public class Application implements CommandLineRunner, ExitCodeGenerator { private int exitCode; private final FileToolsCommand fileToolsCommand; private final IFactory factory; public Application(FileToolsCommand fileToolsCommand, IFactory factory) { this.fileToolsCommand = fileToolsCommand; this.factory = factory; } public static void main(String[] args) { System.exit(SpringApplication.exit(SpringApplication.run(Application.class, args))); } @Override public void run(String... args) { exitCode = new CommandLine(fileToolsCommand, factory) .setExitCodeExceptionMapper(fileToolsCommand) .execute(args); } @Override public int getExitCode() { return exitCode; } }
gradle の build タスクを実行して build/libs の下に nosubcmd-cmdapp-1.0.0-RELEASE.jar を生成します。
Git for Windows の bash を起動してから D:\project-springboot\ksbysample-boot-miscellaneous\picocli-boot-cmdapp\nosubcmd-cmdapp\build\libs\
の下に移動し、alias filetools='java -jar nosubcmd-cmdapp-1.0.0-RELEASE.jar'
コマンドを実行して filetools
だけでコマンドを実行できるようにします。
filetools
コマンドを実行してみると、
Spring Boot のロゴと INFO ログが邪魔でした。。。 今回は表示されないようにします。
src/main/resources/application.properties に以下の内容を記載します。
spring.main.banner-mode=off logging.level.root=OFF
build し直してから filetools
コマンドを実行すると -c, -d のどちらのオプションを指定していないというエラーメッセージ(Error: Missing required argument (specify one of these): ([-c] | [-d])
)とヘルプが表示されました。あれだけの記述なのにヘルプは見やすいし、オプションのところに色が付いているのがいいですね。
-c, -d のオプションをどちらも指定すると Error: --create, --delete are mutually exclusive (specify only one)
というエラーメッセージが表示されます。
filetools -h
コマンドを実行するとヘルプだけが表示され、filetools -V
コマンドを実行するとバージョン番号だけが表示されます。
ファイルの作成、削除を試してみます。ディレクトリ内に jar 以外のファイルがない状態で、
filetools -c 1.txt 2.txt 3.txt
コマンドを実行すると、
ディレクトリ内にファイルが作成されます。
filetools -d 1.txt 2.txt 3.txt
コマンドを実行すると、
ファイルが削除されます。
filetools -c a.txt a.txt
コマンドを実行して同じファイルを2度作成しようとすると、コマンドの戻り値が 0 ではなく 12 になります。
Spring Boot ベースのコマンドラインアプリケーションのサンプルを作成する(Subcommand あり)
今度は Subcommand ありのコマンドラインアプリケーションを作成してみます。Git コマンドでの git commit ...
や git branch ...
のように commit, branch が Subcommand にあたります。
以下の仕様のコマンドを作成します。
cal add 数値 数値 ...
で指定された数値を全て加算した結果を表示する。--avg
('-a' でも可)オプションを付けると数値の個数で割った平均値を表示する。cal multi 数値 数値 ...
で指定された数値を全て乗算した結果を表示する。--compare 数値
オプションを付けると計算結果と数値を比較して、計算結果 < 数値なら -1、計算結果 = 数値なら 0、計算結果 > 数値なら 1 を返す。
まず D:\project-springboot\ksbysample-boot-miscellaneous\picocli-boot-cmdapp
の下に Spring Initializr で subcmd-cmdapp プロジェクトを作成した後、nosubcmd-cmdapp プロジェクトの build.gradle をコピーします。
src/main/java/ksbysample/cmdapp/subcmd の下に CalCommand.java を新規作成して、以下の内容を記載します。
package ksbysample.cmdapp.subcmd; import org.springframework.stereotype.Component; import picocli.CommandLine.*; import java.math.BigDecimal; import java.util.Arrays; import java.util.Optional; import java.util.concurrent.Callable; @Component @Command(name = "cal", mixinStandardHelpOptions = true, versionProvider = CalCommand.class, description = "渡された数値の加算・乗算を行うツール", subcommands = { CalCommand.AddCommand.class, CalCommand.MultiCommand.class }) public class CalCommand implements Callable<Integer>, IExitCodeExceptionMapper, IVersionProvider { @Override public Integer call() { return ExitCode.OK; } @Override public int getExitCode(Throwable exception) { Throwable cause = exception.getCause(); if (cause instanceof NumberFormatException) { // 数値パラメータに数値以外の文字が指定された return 12; } return 11; } @Override public String[] getVersion() { return new String[]{"1.0.0"}; } @Component @Command(name = "add", mixinStandardHelpOptions = true, versionProvider = CalCommand.class, description = "渡された数値を加算する") static class AddCommand implements Callable<Integer> { @Option(names = {"-a", "--avg"}, description = "平均値を算出する") private boolean optAvg; @Parameters(paramLabel = "数値", arity = "1..*", description = "加算する数値") private BigDecimal[] nums; @Override public Integer call() { BigDecimal sum = Arrays.asList(nums).stream() .reduce(new BigDecimal("0"), (a, v) -> a.add(v)); Optional<BigDecimal> avg = optAvg ? Optional.of(sum.divide(BigDecimal.valueOf(nums.length))) : Optional.empty(); System.out.println(avg.orElse(sum)); return ExitCode.OK; } } @Component @Command(name = "multi", mixinStandardHelpOptions = true, versionProvider = CalCommand.class, description = "渡された数値を乗算する") static class MultiCommand implements Callable<Integer> { @Parameters(paramLabel = "数値", arity = "1..*", description = "乗算する数値") private BigDecimal[] nums; @Option(names = {"-c", "--compare"}, description = "計算結果と比較して、計算結果 < 数値なら -1、計算結果 = 数値なら 0、計算結果 > 数値なら 1 を返す") private BigDecimal compareNum; @Override public Integer call() { BigDecimal result = Arrays.asList(nums).stream() .reduce(new BigDecimal("1"), (a, v) -> a.multiply(v)); Optional<Integer> compareResult = (compareNum == null) ? Optional.empty() : Optional.of(result.compareTo(compareNum)); System.out.println(compareResult.isPresent() ? compareResult.get() : result); return ExitCode.OK; } } }
src/main/java/ksbysample/cmdapp/subcmd/Application.java を以下のように変更します。
package ksbysample.cmdapp.subcmd; import org.springframework.boot.CommandLineRunner; import org.springframework.boot.ExitCodeGenerator; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import picocli.CommandLine; import picocli.CommandLine.ExitCode; import picocli.CommandLine.IFactory; import picocli.CommandLine.ParameterException; import picocli.CommandLine.ParseResult; @SpringBootApplication public class Application implements CommandLineRunner, ExitCodeGenerator { private int exitCode; private final CalCommand calCommand; private final IFactory factory; public Application(CalCommand calCommand , IFactory factory) { this.calCommand = calCommand; this.factory = factory; } public static void main(String[] args) { System.exit(SpringApplication.exit(SpringApplication.run(Application.class, args))); } @Override public void run(String... args) { CommandLine commandLine = new CommandLine(calCommand, factory); // subcommand が指定されていない場合にはエラーメッセージと usage を表示する try { ParseResult parsed = commandLine.parseArgs(args); if (parsed.subcommand() == null && !parsed.isUsageHelpRequested() && !parsed.isVersionHelpRequested()) { System.err.println("Error: at least 1 command and 1 subcommand found."); commandLine.usage(System.out); exitCode = ExitCode.USAGE; return; } } catch (ParameterException ignored) { // CommandLine#parseArgs で ParameterException が throw されても // CommandLine#execute を実行しないと subcommand の usage が表示されないので // ここでは何もしない } exitCode = commandLine .setExitCodeExceptionMapper(calCommand) .execute(args); } @Override public int getExitCode() { return exitCode; } }
src/main/resources/application.properties を以下のように変更します。
spring.main.banner-mode=off logging.level.root=OFF
bash で D:\project-springboot\ksbysample-boot-miscellaneous\picocli-boot-cmdapp\subcmd-cmdapp\build\libs\
の下に移動し、alias cal='java -jar subcmd-cmdapp-1.0.0-RELEASE.jar'
コマンドを実行して cal
だけでコマンドを実行できるようにします。
cal
コマンドだけを実行すると Subcommand が指定されていないので Error: at least 1 command and 1 subcommand found.
のエラーメッセージと usage が表示されます。
cal add
コマンドを実行すると、
cal multi
コマンドを実行すると、
--version(-V)オプション指定時に build.gradle に記述した build.version を表示する
build.gradle に version を記述しているので、cal -V
実行時にこの文字列を出力するようにしてみます。
buildscript { ext { group "ksby.ksbysample-boot-miscellaneous.picocli-boot-cmdapp" version "1.0.0-RELEASE" }
まず build.gradle に springBoot { buildInfo() }
を追加します。
`
idea { module { inheritOutputDirs = false outputDir = file("$buildDir/classes/main/") } } springBoot { buildInfo() } repositories { mavenCentral() }
src/main/java/ksbysample/cmdapp/subcmd/CalCommand.java を以下のように変更します。
public class CalCommand implements Callable<Integer>, IExitCodeExceptionMapper, IVersionProvider { // picocli.AutoComplete で generate Completion Script をする時に引数なしのコンストラクタが必要になる // のでコンストラクタインジェクションは使用しないこと // https://picocli.info/autocomplete.html 参照 @Autowired private BuildProperties buildProperties; .......... @Override public String[] getVersion() { return new String[]{buildProperties.getVersion()}; } ..........
private final BuildProperties buildProperties;
を追加します。- getVersion メソッド内で
"1.0.0"
→buildProperties.getVersion()
に変更します。
build して jar ファイルを作成し直してから cal -V
コマンドを実行すると build.gradle の version に記述した文字列が表示されます。
TAB キー押下時に subcommand, option の候補の表示や自動補完が行われるようにする
Autocomplete for Java Command Line Applications のマニュアルに従い、cal
コマンドの subcommand, option の候補の表示や自動補完が行わえるようにしてみます。
subcmd-cmdapp-1.0.0-RELEASE.jar を zip 解凍可能なツール(今回は Explzh を使用)で開いた後、BOOT-INF/lib の下にある spring-boot-2.1.6.RELEASE.jar と picocli-4.0.0.jar を取り出します。
java -cp "picocli-4.0.0.jar;subcmd-cmdapp-1.0.0-RELEASE.jar" picocli.AutoComplete -n cal ksbysample.cmdapp.subcmd.CalCommand
を実行しても ClassNotFoundException が発生して動作しなかったのですが、
class ファイルは subcmd-cmdapp/build/classes/java/main の下に生成されているので、
java -cp "picocli-4.0.0.jar;spring-boot-2.1.6.RELEASE.jar;../classes/java/main" picocli.AutoComplete -n cal ksbysample.cmdapp.subcmd.CalCommand
を実行します。
cal_completion
というファイルが生成されます。
何もしていない時には bash 上で cal
と入力してから TAB キーを2回押すととディレクトリ内のファイル一覧が表示されますが、
. cal_completion
コマンドを実行してから cal
を入力+TAB キーを2回押すと subcommand の候補が表示されます。
cal m
とだけ入力して TABキーを1回押すと、
multi
の文字列が自動補完されます。
cal multi -
と入力してから TAB キーを2回押すと指定可能なオプションが表示されますし、
cal multi --c
と入力してから TAB キーを1回押すと
--compare
のオプションが自動補完されます。
履歴
2019/07/20
初版発行。