aboutsummaryrefslogtreecommitdiffstats
path: root/infrastructure/net.appjet.common.cli/cli.scala
blob: ef9223fef0c063c66ad2aaf5b6f8bfc370608591 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
/**
 * Copyright 2009 Google Inc.
 * 
 * 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
 * 
 *      http://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 net.appjet.common.cli;

import org.apache.commons.lang.WordUtils;

class CliOption(val name: String, val description: String, val argName: Option[String]);

class ParseException(message: String) extends RuntimeException(message);

class CliParser(predef: Array[CliOption]) {
  val displayWidth = 80;
  val options = Map((for (opt <- predef) yield ((opt.name, opt))): _*);

  def parseOptions(args0: Array[String]): (Map[String, String], Array[String]) = {
    val (opts, args) = args0.partition(_.startsWith("-"));
    (Map((for (arg <- opts) yield {
      val parts = arg.split("=", 2);
      val name = "-+".r.replaceFirstIn(parts(0), "");
      if (parts.length == 1 && options.get(name).map(_.argName.isDefined).exists(x => x))
	throw new ParseException("Missing argument for flag: "+name);
      (name, parts.orElse(Map(1 -> "true"))(1));
    }): _*),
     args.toArray);
  }

  def dprint(prefix: String, value: String) = {
//    println(prefix+": "+value+"\n");
    value;
  }

  def usage = {
    val sb = new StringBuilder();
    var maxLength = predef.map(opt => 2 + opt.name.length + opt.argName.map(_.length + 1).getOrElse(0) ).reduceRight(Math.max)+2;
    for ((n, opt) <- options) {
      sb.append("  --"+n+opt.argName.map("=<"+_+">").getOrElse("")+"\n");
      sb.append("     "+WordUtils.wrap(opt.description, displayWidth-5).split("\n").mkString("\n     "));
      sb.append("\n\n");
    }
    sb.toString();
  }
}