001/* 002 * Copyright (c) 2009 The openGion Project. 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, 013 * either express or implied. See the License for the specific language 014 * governing permissions and limitations under the License. 015 */ 016package org.opengion.fukurou.taglet2; 017 018import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE; // 6.1.0.0 (2014/12/26) refactoring 019 020import jdk.javadoc.doclet.DocletEnvironment ; 021// import jdk.javadoc.doclet.Doclet ; 022// import jdk.javadoc.doclet.Reporter ; 023import javax.lang.model.element.Element ; 024import javax.lang.model.element.Modifier ; 025import javax.lang.model.element.TypeElement; 026import javax.lang.model.element.ElementKind ; 027import javax.lang.model.element.VariableElement; 028import javax.lang.model.element.ExecutableElement; 029// import javax.lang.model.SourceVersion ; 030import javax.lang.model.type.TypeMirror; 031import javax.lang.model.type.TypeKind; 032import javax.lang.model.util.ElementFilter ; 033import javax.lang.model.util.Elements ; 034import javax.lang.model.util.Types ; 035import javax.tools.Diagnostic.Kind ; 036import com.sun.source.doctree.DocCommentTree ; 037import com.sun.source.util.DocTrees ; 038// import com.sun.source.util.DocSourcePositions ; 039import com.sun.source.doctree.DocTree ; 040 041// import java.util.Locale ; 042import java.util.Set; 043import java.util.List; 044import java.util.HashSet; 045import java.util.Arrays; 046// import java.util.stream.Stream; // 6.4.3.4 (2016/03/11) 047// import java.util.stream.Collectors; // 6.4.3.4 (2016/03/11) 048// import java.util.regex.Pattern; // 6.3.9.1 (2015/11/27) final化に伴う整理 049 050// import java.io.IOException; 051// import java.io.File; 052// import java.io.PrintWriter; 053import java.util.stream.Stream; // 6.4.3.4 (2016/03/11) 054import java.util.stream.Collectors; // 6.4.3.4 (2016/03/11) 055import java.util.Map; 056 057// import org.opengion.fukurou.util.FileUtil; 058// import org.opengion.fukurou.util.StringUtil; 059 060/** 061 * ソースコメントから、パラメータ情報を取り出す Doclet クラスです。 062 * og.paramLevel タグと og.cryptography タグを切り出します。 063 * これらは、システムパラメータとしてGE12テーブルに設定される値をクラスより抽出する 064 * のに使用します。 065 * 066 * @version 7.3 067 * @author Kazuhiko Hasegawa 068 * @since JDK11.0, 069 */ 070public class DocTreeSpecific extends AbstractDocTree { 071 private static final String SELECT_PACKAGE = "org.opengion" ; 072 private static final boolean USE_PRIVATE = false ; 073 074 private static final String OG_FOR_SMPL = "og.formSample"; 075 private static final String OG_REV = "og.rev"; 076 private static final String OG_GROUP = "og.group"; 077 private static final String DOC_VERSION = "version"; 078 private static final String DOC_AUTHOR = "author"; 079 private static final String DOC_SINCE = "since"; 080 081// private static final String OG_TAG_NAME = "og.tag"; // 6.1.2.0 (2015/01/24) チェック用 082// private static final String DOC_PARAM = "param"; // 5.1.9.0 (2010/08/01) チェック用 083// private static final String DOC_RETURN = "return"; // 5.1.9.0 (2010/08/01) チェック用 084 085 private static final String CONSTRUCTOR = "コンストラクタ" ; 086 private static final String METHOD = "メソッド" ; 087 088 private String version ; 089 private String outfile ; 090 private int debugLevel ; // 0:なし 1:最小チェック 2:日本語化 3:体裁 4:Verチェック 5:taglibラベル 091 092 private final Set<String> mtdClsSet = new HashSet<>(); 093 094 // 5.1.9.0 (2010/08/01) ソースチェック用(半角文字+空白文字のみ) 095// private static final Pattern PTN = Pattern.compile("[\\w\\s]+"); // 6.3.9.1 (2015/11/27) 096 097// private DocTrees docUtil; 098// private Elements eleUtil ; 099 private Types typUtil ; 100 101 /** 102 * Doclet のエントリポイントメソッドです(昔の startメソッド)。 103 * 104 * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 105 * 106 * @param docEnv ドックレットを1回呼び出す操作環境 107 * 108 * @return 正常実行時 true 109 */ 110 @Override 111 public boolean run( final DocletEnvironment docEnv ) { 112 try( DocTreeWriter writer = new DocTreeWriter( outfile,ENCODE ) ) { 113 writer.printTag( "<?xml version=\"1.0\" encoding=\"", ENCODE , "\" ?>" ); 114 writer.printTag( "<javadoc>" ); 115 writer.printTag( " <version>",version,"</version>" ); 116 writer.printTag( " <description></description>" ); 117 writeContents( docEnv,writer ); 118 writer.printTag( "</javadoc>" ); 119 } 120 catch( final Throwable th ) { 121 reporter.print(Kind.ERROR, th.getMessage()); 122 } 123 124 return true; 125 } 126 127 /** 128 * DocletEnvironmentよりコンテンツを作成します。 129 * 130 * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 131 * 132 * @param docEnv ドックレットの最上位 133 * @param writer DocTreeWriterオブジェクト 134 */ 135 private void writeContents( final DocletEnvironment docEnv, final DocTreeWriter writer ) { 136// docUtil = docEnv.getDocTrees(); 137 final DocTrees docUtil = docEnv.getDocTrees(); 138 final Elements eleUtil = docEnv.getElementUtils(); 139 typUtil = docEnv.getTypeUtils(); 140 141 // クラス単位にループする。 142 for( final TypeElement typEle : ElementFilter.typesIn(docEnv.getIncludedElements())) { 143 final String fullName = String.valueOf( typEle.getQualifiedName() ) ; 144 writer.setClassName( fullName ); 145 146 final String className = String.valueOf( typEle.getSimpleName() ); 147 148 final StringBuilder modiBuf = new StringBuilder(); 149 typEle.getModifiers().forEach( modi -> modiBuf.append( modi ).append( ' ' ) ); 150 final ElementKind eleKind = typEle.getKind(); 151 if( ElementKind.CLASS.equals( eleKind ) ) { modiBuf.append( "class" ); } 152 final String modifiers = modiBuf.toString().trim(); 153 154 final TypeMirror superType = typEle.getSuperclass(); 155 final String superClass = TypeKind.NONE.equals( superType.getKind() ) ? "" : superType.toString(); 156 157 final String intFase = Stream.of( typEle.getInterfaces() ) 158 .map( typ -> String.valueOf(typ) ) 159 .collect( Collectors.joining( "," ) ); 160 161 final DocCommentTree docTree = docUtil.getDocCommentTree(typEle); // ドキュメンテーション・コメントが見つからない場合、null が返る。 162 163 final List<? extends DocTree> desc = docTree == null ? EMPTY_LIST : docTree.getFirstSentence(); 164 final List<? extends DocTree> cmnt = docTree == null ? EMPTY_LIST : docTree.getFullBody(); 165 166 final Map<String,List<String>> blkTagMap = blockTagsMap(docTree); 167 final String smplTags = getBlockTag( OG_FOR_SMPL, blkTagMap, "" ); 168 final String revTags = getBlockTag( OG_REV , blkTagMap, "\n" ); 169 final String createVer = getBlockTag( DOC_VERSION, blkTagMap, "" ); 170 final String author = getBlockTag( DOC_AUTHOR , blkTagMap, "" ); 171 final String since = getBlockTag( DOC_SINCE , blkTagMap, "" ); 172 final String grpTags = getBlockTag( OG_GROUP , blkTagMap, "," ); 173 174 // 5.7.1.1 (2013/12/13) タグのインデントを止める。 175 writer.printTag( "<classDoc>" ); 176 writer.printTag( " <fullName>" ,fullName ,"</fullName>" ); 177 writer.printTag( " <modifiers>" ,modifiers ,"</modifiers>" ); 178 writer.printTag( " <className>" ,className ,"</className>" ); 179 writer.printTag( " <superClass>" ,superClass ,"</superClass>" ); 180 writer.printTag( " <interface>" ,intFase ,"</interface>" ); 181 writer.printTag( " <createVer>" ,createVer ,"</createVer>" ); 182 writer.printTag( " <author>" ,author ,"</author>" ); 183 writer.printTag( " <since>" ,since ,"</since>" ); 184 writer.printTag( " <description>" ,desc ,"</description>" ); 185 writer.printTag( " <contents>" ,cmnt ,"</contents>" ); 186 writer.printTag( " <classGroup>" ,grpTags ,"</classGroup>" ); 187 writer.printTag( " <formSample>" ,smplTags ,"</formSample>" ); 188 writer.printTag( " <history>" ,revTags ,"</history>" ); 189 190 191 // 5.1.9.0 (2010/08/01) ソースチェック用(コメントや概要が無い場合。スーパークラスは省く) 192 if( debugLevel >= 2 && ( cmnt.isEmpty() || desc.isEmpty() ) && superClass.isEmpty() ) { 193 // final DocSourcePositions srcPos = docUtil.getSourcePositions(); 194 // System.err.println( "警告2:コメントC=\t" + srcPos ); 195 System.err.println( "②警告2:コメントC=\t" + typEle ); 196 } 197 198 int extendFlag = 0; // 0:オリジナル 1:org.opengion関連Extend 2:Java関連Extend 199 200 // 6.4.3.0 (2016/02/05) PMDチェックのDocletでのフォロー。 201 // checkPMD( typEle ); 202 203 // 5.6.6.0 (2013/07/05) VERSION staticフィールドと、@og.rev コメントの比較チェック 204 // while 以下で、fullName と classDoc を順番に上にさかのぼっているので、先にチェックします。 205 // checkTag2( typEle ); 206 207 mtdClsSet.clear(); 208 TypeElement loopEle = typEle; 209 String superName = fullName; 210 while( loopEle != null ) { 211 writer.setClassName( superName ); 212 213 // 対象クラスの スーパークラスを取得しておく(メソッド内で使うのと、上位に上がってく時に使う) 214 final TypeMirror spType = loopEle.getSuperclass(); 215 216 // AbstractObjectPool<java.sql.Connection> の様な型を削除します。 217 final StringBuilder buf = new StringBuilder().append(spType); 218 int st = buf.indexOf("<"); 219 int ed = buf.indexOf(">",st); 220 while( st > 0 ) { 221 buf.delete(st,ed+1); 222 st = buf.indexOf("<"); 223 ed = buf.indexOf(">",st); 224 } 225 superName = buf.toString(); 226 // superName = String.valueOf( spType ); 227 228 final String extClass = ( extendFlag == 0 ) ? "" : superName ; 229 230 // コンストラクタのみフィルタリングして取得する 231 for( final ExecutableElement exEle : ElementFilter.constructorsIn(loopEle.getEnclosedElements()) ) { 232 final DocCommentTree dct = docUtil.getDocCommentTree(exEle); // ドキュメンテーション・コメントが見つからない場合、null が返る。 233 if( dct != null && isAction( exEle,extendFlag ) ) { 234 // checkTag( exEle,dct ); // 5.5.4.1 (2012/07/06) チェックを分離 235 menberTag( exEle,dct,CONSTRUCTOR,writer,extendFlag,extClass ); 236 } 237 } 238 239 // メソッドのみフィルタリングして取得する 240 for( final ExecutableElement exEle : ElementFilter.methodsIn(loopEle.getEnclosedElements())) { 241 final DocCommentTree dct = docUtil.getDocCommentTree(exEle); // ドキュメンテーション・コメントが見つからない場合、null が返る。 242 if( dct != null && isAction( exEle,extendFlag ) ) { 243 // checkTag( exEle,dct ); 244 menberTag( exEle,dct,METHOD,writer,extendFlag,extClass ); 245 } 246 } 247 248 // 対象クラス(オリジナル)から、上に上がっていく。 249 if( superName.startsWith( SELECT_PACKAGE ) ) { 250 extendFlag = 1; 251 } 252 else { 253 break; 254 } 255 256 loopEle = eleUtil.getTypeElement(superName); 257 } 258 writer.printTag( "</classDoc>" ); 259 } 260 } 261 262 /** 263 * メンバークラスのXML化を行うかどうか[true/false]を判定します。 264 * 265 * 以下の条件に合致する場合は、処理を行いません。(false を返します。) 266 * 267 * 1.同一クラスを処理中にEXTENDで継承元をさかのぼる場合、すでに同じシグネチャのメソッドが 268 * 存在している。 269 * 2.USE_PRIVATE が true の時の private メソッド 270 * 3.extendFlag が 0以上(1,2)の時の private メソッド 271 * 4.メソッド名におかしな記号(<など)が含まれている場合 272 * 273 * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 274 * 275 * @param menber ExecutableMemberDocオブジェクト 276 * @param extendFlag 継承状態 [0:オリジナル/1:org.opengion関連Extend/2:Java関連Extend] 277 * 278 * @return XML化を行うかどうか[true/false] 279 */ 280 private boolean isAction( final ExecutableElement menber,final int extendFlag ) { 281// final String menberName = String.valueOf( menber.getSimpleName() ) ; 282 final boolean isPrivate = menber.getModifiers().contains( Modifier.PRIVATE ); 283 final boolean isTypeParam = ! menber.getTypeParameters().isEmpty(); 284 final boolean rtn = ! mtdClsSet.add( String.valueOf( menber ) ) // 5.5.4.1 (2012/07/06) メソッドの重複処理判定は、クラス名も含めて行う 285 || USE_PRIVATE && isPrivate 286 || extendFlag > 0 && isPrivate 287 || isTypeParam; // PMD Useless parentheses. 288 289 return ! rtn ; 290 } 291 292// /** 293// * param,return 等の整合性をチェックします。 294// * 295// * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 296// * 297// * @param exEle ExecutableElementオブジェクト 298// * @param menber DocCommentTreeオブジェクト 299// */ 300// private void checkTag( final ExecutableElement exEle , final DocCommentTree menber ) { 301// // 未実装 302// } 303 304// /** 305// * PMDで、チェックしている処理のうち、Docletでフォローできる分をチェックします。 306// * 307// * ※ このチェックは、警告レベル5 のみ集約していますので、呼出元で、制限します。 308// * 309// * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 310// * 311// * @param typEle TypeElementオブジェクト 312// */ 313// private void checkTag2( final TypeElement typEle ) { 314// String cnstVar = null ; // 初期値 315// String seriUID = null ; 316// final String src = "\tsrc/" + String.valueOf(typEle).replace('.','/') + ".java:100" ; // 行が判らないので、100行目 決め打ち 317// 318// // フィールドのみフィルタリングして取得する 319// for( final VariableElement varEle : ElementFilter.fieldsIn(typEle.getEnclosedElements())) { // フィールドだけに絞る 320// final String key = String.valueOf(varEle); 321// if( "VERSION".equals( key ) ) { 322// cnstVar = String.valueOf( varEle.getConstantValue() ); 323// } 324// else if( "serialVersionUID".equals( key ) ) { 325// seriUID = String.valueOf( varEle.getConstantValue() ) + "L"; // 旧JavaDocと違い、"L" まで取ってこれないみたい 326// } 327// } 328// 329// if( cnstVar == null ) { return; } // VERSION が未定義のクラスは処理しない 330// 331// String maxRev = cnstVar ; // 5.7.1.1 (2013/12/13) 初期値 332// boolean isChange = false; // max が入れ替わったら、true 333// 334// // メソッドのみフィルタリングして取得する 335// for( final ExecutableElement exEle : ElementFilter.methodsIn(typEle.getEnclosedElements())) { 336// final DocCommentTree dct = docUtil.getDocCommentTree(exEle); // ドキュメンテーション・コメントが見つからない場合、null が返る。 337// final Map<String,List<String>> blkTagMap = blockTagsMap(dct); 338// final List<String> revTags = blkTagMap.get("og.rev"); 339// 340// if( revTags != null ) { 341// for( final String tag :revTags ) { // 複数存在しているはず 342// final int idx = tag.indexOf( ' ' ); // 最初のスペース 343// if( idx > 0 ) { 344// final String rev = tag.substring( 0,idx ).trim(); 345// if( maxRev.compareTo( rev ) < 0 ) { // revTags の og.rev が大きい場合 346// maxRev = rev ; 347// isChange = true; 348// } 349// } 350// } 351// } 352// } 353// 354// // VERSION 文字列 の定義があり、かつ、max の入れ替えが発生した場合のみ、警告4:VERSIONが古い 355// if( isChange ) { // 5.7.1.1 (2013/12/13) 入れ替えが発生した場合 356// System.err.println( "警告4:VERSIONが古い=\t" + cnstVar + " ⇒ " + maxRev + src ); 357// } 358// 359// // serialVersionUID の定義がある。 360// if( seriUID != null ) { 361// final StringBuilder buf = new StringBuilder(); 362// // maxRev は、最大の Revか、初期のVERSION文字列 例:5.6.6.0 (2013/07/05) 363// for( int i=0; i<maxRev.length(); i++ ) { // 364// final char ch = maxRev.charAt( i ); 365// if( ch >= '0' && ch <= '9' ) { buf.append( ch ); } // 数字だけ取り出す。 例:566020130705 366// } 367// buf.append( 'L' ); // 強制的に、L を追加する。 368// final String maxSeriUID = buf.toString() ; 369// 370// // 5.7.1.1 (2013/12/13) 値の取出し。Long型を表す "L" も含まれている。 371// if( !maxSeriUID.equals( seriUID ) ) { // 一致しない 372// System.err.println( "警告4:serialVersionUIDが古い=\t" + seriUID + " ⇒ " + maxSeriUID + src ); 373// } 374// } 375// } 376 377// /** 378// * PMDで、チェックしている処理のうち、Docletでフォローできる分をチェックします。 379// * 380// * ※ このチェックは、警告レベル5 のみ集約していますので、呼出元で、制限します。 381// * 382// * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 383// * 384// * @param typEle TypeElementオブジェクト 385// */ 386// private void checkPMD( final TypeElement typEle ) { 387// // 未実装 388// } 389 390 /** 391 * メンバークラス(コンストラクタ、メソッド)をXML化します。 392 * 393 * @og.rev 7.3.0.0 (2021/01/06) 新しいJavaDoc対応 394 * 395 * @param menber ExecutableElementオブジェクト 396 * @param docTree DocCommentTreeオブジェクト 397 * @param menberType メンバータイプ(コンストラクタ、メソッド) 398 * @param writer Tagを書き出すWriterオブジェクト 399 * @param extendFlag 継承状態 [0:オリジナル/1::org.opengion関連Extend/2:Java関連Extend] 400 * @param extClass 継承クラス(オリジナルの場合は、空文字列) 401 */ 402 private void menberTag( final ExecutableElement menber, 403 final DocCommentTree docTree , 404 final String menberType, 405 final DocTreeWriter writer, 406 final int extendFlag , 407 final String extClass ) { 408 409 final String modifiers ; 410 411 final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE ); 412 menber.getModifiers().forEach( modi -> buf.append( modi ).append( ' ' ) ); 413 if( ElementKind.METHOD.equals( menber.getKind() ) ) { // メソッドの処理。コンストラクタの場合は、返り値がないので処理しない。 414 buf.append( menber.getReturnType() ); 415 } 416 modifiers = buf.toString(); 417 418 final String menberName = String.valueOf( menber.getSimpleName() ); // コンストラクタの場合は、<init> が返る。 419 420 final StringBuilder sigBuf = new StringBuilder().append( menberName ).append( '(' ) ; 421 boolean flag = false; 422 for( final VariableElement valEle : menber.getParameters() ) { 423 flag = true; 424 final Element ele = typUtil.asElement( valEle.asType() ); // 型が対応する要素を持たない場合はnullを返します。 425 final String key = ele == null // 配列や、プリミティブ型の場合は、null になっている。 426 ? String.valueOf( valEle.asType() ) 427 : String.valueOf( ele.getSimpleName() ); 428 429 sigBuf.append( key ).append( ' ' ) 430 .append( valEle.getSimpleName() ).append( ',' ); 431 } 432 433 if( flag ) { sigBuf.deleteCharAt( sigBuf.length()-1 ); } 434 sigBuf.append( ')' ); 435 final String signature = sigBuf.toString(); 436// final String signature = String.valueOf( menber ); 437 438 final List<? extends DocTree> desc = docTree == null ? EMPTY_LIST : docTree.getFirstSentence(); 439 final List<? extends DocTree> cmnt = docTree == null ? EMPTY_LIST : docTree.getFullBody(); 440 441 final Map<String,List<String>> blkTagMap = blockTagsMap(docTree); 442 final String revTags = getBlockTag( OG_REV, blkTagMap, "\n" ); 443 444 // tags は、OG_REV 以外のすべてで、かつ、キーワードも含む。 445 final StringBuilder tagBuf = new StringBuilder(); 446 if( docTree != null ) { 447 for( final DocTree dt : docTree.getBlockTags() ) { 448 final String tag = String.valueOf(dt).trim(); 449 if( !tag.contains( OG_REV ) ) { tagBuf.append( tag ).append( '\n' ); } 450 } 451 } 452 final String tags = tagBuf.toString().trim(); 453 454// final StringBuilder tagBuf = new StringBuilder(); 455// final StringBuilder revBuf = new StringBuilder(); 456// if( docTree != null ) { 457// for( final DocTree dt : docTree.getBlockTags() ) { 458// final String tag = String.valueOf(dt).trim(); 459// if( tag.contains( OG_REV ) ) { revBuf.append( cutTag( tag,OG_REV ) ).append( '\n' ); } 460// else { tagBuf.append( tag ).append( '\n' ); } 461// } 462// } 463// final String revTags = revBuf.toString().trim(); 464// final String tags = tagBuf.toString().trim(); 465 466 final String extend = String.valueOf( extendFlag ); 467 468 // final DocSourcePositions srcPos = docUtil.getSourcePositions(); 469 // final String position = String.valueOf( srcPos.getStartPosition( null,docTree,null ) ); 470 final String position = ""; 471 472 writer.printTag( " <menber>" ); 473 writer.printTag( " <type>" ,menberType ,"</type>" ); 474 writer.printTag( " <name>" ,menberName ,"</name>" ); 475 writer.printTag( " <modifiers>" ,modifiers ,"</modifiers>" ); 476 writer.printTag( " <signature>" ,signature ,"</signature>" ); 477 writer.printTag( " <position>" ,position ,"</position>" ); 478 writer.printTag( " <extendClass>",extClass ,"</extendClass>" ); 479 writer.printTag( " <extendFlag>" ,extend ,"</extendFlag>" ); 480 writer.printTag( " <description>",desc ,"</description>" ); 481 writer.printTag( " <contents>" ,cmnt ,"</contents>" ); 482 writer.printTag( " <tagText>" ,tags ,"</tagText>" ); 483 writer.printTag( " <history>" ,revTags ,"</history>" ); 484 writer.printTag( " </menber>"); 485 } 486 487 /** 488 * サポートされているすべてのオプションを返します。 489 * 490 * @return サポートされているすべてのオプションを含むセット、存在しない場合は空のセット 491 */ 492 @Override 493 public Set<? extends Option> getSupportedOptions() { 494 final Option[] options = { 495 new AbstractOption( "-outfile", "-version", "-debugLevel" ) { 496 497 /** 498 * 必要に応じてオプションと引数を処理します。 499 * 500 * @param opt オプション名 501 * @param arguments 引数をカプセル化したリスト 502 * @return 操作が成功した場合はtrue、そうでない場合はfalse 503 */ 504 @Override 505 public boolean process(final String opt, final List<String> arguments) { 506 if( "-outfile".equalsIgnoreCase(opt) ) { 507 outfile = arguments.get(0); 508 } 509 else if( "-version".equalsIgnoreCase(opt) ) { 510 version = arguments.get(0); 511 } 512 else if( "-debugLevel".equalsIgnoreCase(opt) ) { 513 debugLevel = Integer.parseInt( arguments.get(0) ); 514 } 515 return true; 516 } 517 } 518 }; 519 return new HashSet<>(Arrays.asList(options)); 520 } 521}