2016-11-04 8 views
0

、私は、関数名とそのパラメータを受け入れ、解析し、タブ補完とパーサを書きたい:SBTではNoSuchElementException依存パーサ

case class Param(name : String, tpe : String) // tpe is for param type 
case class Func(name : String, params : immutable.Seq[Param]) 

def funcAndParamsParser(funcs : immutable.Set[Func]) : Parser[(Func,immutable.Seq[String])] = ??? 

はこれを実現するために、私はにそれに問題を破ります作品:

def funcParser(funcs : immutable.Set[Func]) : Parser[Func] = { 
    val funcsByName = funcs.map(func => (func.name, func)).toMap 
    NotSpace.examples(funcsByName.keySet).map(funcsByName) 
} 

def paramParser(param : Param) : Parser[String] = { 
    NotSpace.examples(s"<${param.name}, of type ${param.tpe}>") 
} 

def paramsParser(params : immutable.Seq[Param]) : Parser[immutable.Seq[String]] = { 
    params.map(paramParser).foldLeft(success(immutable.Seq.empty[String])){ 
    (nascent, next) => nascent.flatMap(partial => Space ~> next.map(str => partial :+ str)) 
    } 
} 

最後に、私はメインイベントを実装することができます。

def funcAndParamsParser(funcs : immutable.Set[Func]) : Parser[(Func,immutable.Seq[String])] = { 
    funcParser(funcs).flatMap(func => paramsParser(func.params).map(seq => (func, seq))) 
} 

すべての子パーサーは正常に動作しますが、最後の組み合わせはfuncAndParamsParser(...)で、NoSuchElementExceptionで失敗します。

val func0 = Func("move", Param("x", "Int") :: Param("y", "Int") :: Nil) 
val func1 = Func("fill", Param("color", "Color") :: Nil) 
val testParser = Space ~> funcAndParamsParser(immutable.Set(func0, func1)) 

私はtestParser.parsedを呼び出しInputTaskでこのパーサを埋め込むと、一種の動作しているようです。タスク名の後に<space>と入力して<tab>を入力すると、期待される関数のリストが表示されます(ただし、第2引数の例としてはmoveは省略されています)。

> tmpTestDynamicParser 
fill <color, of type Color> move <x, of type Int> 

ただし、パーサーは常に最終的に失敗します。私は、有効な入力がどうあるべきかで入力した場合:

java.util.NoSuchElementException: key not found: m 
    at scala.collection.MapLike$class.default(MapLike.scala:228) 
    at scala.collection.AbstractMap.default(Map.scala:58) 
    at scala.collection.MapLike$class.apply(MapLike.scala:141) 
    at scala.collection.AbstractMap.apply(Map.scala:58) 
    at sbt.complete.Parser$Value.map(Parser.scala:161) 
    at sbt.complete.MapParser.resultEmpty$lzycompute(Parser.scala:704) 
    at sbt.complete.MapParser.resultEmpty(Parser.scala:704) 
    at sbt.complete.BindParser.derive(Parser.scala:694) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.ParserSeq$$anonfun$derive$6.apply(Parser.scala:676) 
    at sbt.complete.ParserSeq$$anonfun$derive$6.apply(Parser.scala:676) 
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:244) 
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:244) 
    at scala.collection.immutable.List.foreach(List.scala:318) 
    at scala.collection.TraversableLike$class.map(TraversableLike.scala:244) 
    at scala.collection.AbstractTraversable.map(Traversable.scala:105) 
    at sbt.complete.ParserSeq.derive(Parser.scala:676) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.MapParser.derive(Parser.scala:705) 
    at sbt.complete.ParserMain$class.derive1(Parser.scala:454) 
    at sbt.complete.Parser$.derive1(Parser.scala:135) 
    at sbt.complete.ParserMain$$anonfun$apply$9.apply(Parser.scala:450) 
    at sbt.complete.ParserMain$$anonfun$apply$9.apply(Parser.scala:450) 
    at scala.collection.IndexedSeqOptimized$class.foldl(IndexedSeqOptimized.scala:51) 
    at scala.collection.IndexedSeqOptimized$class.foldLeft(IndexedSeqOptimized.scala:60) 
    at scala.collection.immutable.StringOps.foldLeft(StringOps.scala:31) 
    at scala.collection.TraversableOnce$class.$div$colon(TraversableOnce.scala:138) 
    at scala.collection.immutable.StringOps.$div$colon(StringOps.scala:31) 
    at sbt.complete.ParserMain$class.apply(Parser.scala:450) 
    at sbt.complete.Parser$.apply(Parser.scala:135) 
    at sbt.complete.ParserMain$class.completions(Parser.scala:464) 
    at sbt.complete.Parser$.completions(Parser.scala:135) 
    at sbt.complete.JLineCompletion$$anonfun$parserAsCompletor$1.apply(JLineCompletion.scala:52) 
    at sbt.complete.JLineCompletion$$anonfun$parserAsCompletor$1.apply(JLineCompletion.scala:52) 
    at sbt.complete.JLineCompletion$$anonfun$customCompletor$1$$anonfun$2.apply(JLineCompletion.scala:75) 
    at sbt.complete.JLineCompletion$$anonfun$customCompletor$1$$anonfun$2.apply(JLineCompletion.scala:75) 
    at sbt.complete.JLineCompletion$.complete(JLineCompletion.scala:94) 
    at sbt.complete.JLineCompletion$$anonfun$customCompletor$1.apply(JLineCompletion.scala:75) 
    at sbt.complete.JLineCompletion$$anonfun$customCompletor$1.apply(JLineCompletion.scala:74) 
    at sbt.complete.JLineCompletion$CustomHandler.complete(JLineCompletion.scala:30) 
    at jline.console.ConsoleReader.complete(ConsoleReader.java:3311) 
    at jline.console.ConsoleReader.readLine(ConsoleReader.java:2646) 
    at jline.console.ConsoleReader.readLine(ConsoleReader.java:2372) 
    at jline.console.ConsoleReader.readLine(ConsoleReader.java:2360) 
    at sbt.JLine.sbt$JLine$$readLineDirectRaw(LineReader.scala:42) 
    at sbt.JLine$$anonfun$readLineDirect$2.apply(LineReader.scala:34) 
    at sbt.JLine$$anonfun$readLineDirect$2.apply(LineReader.scala:34) 
    at sbt.Signals0.withHandler(Signal.scala:81) 
    at sbt.Signals$.withHandler(Signal.scala:11) 
    at sbt.JLine.readLineDirect(LineReader.scala:34) 
    at sbt.JLine.readLineWithHistory(LineReader.scala:27) 
    at sbt.JLine.sbt$JLine$$unsynchronizedReadLine(LineReader.scala:19) 
    at sbt.JLine$$anonfun$readLine$1.apply(LineReader.scala:16) 
    at sbt.JLine$$anonfun$readLine$1.apply(LineReader.scala:16) 
    at sbt.JLine$$anonfun$withJLine$1.apply(LineReader.scala:117) 
    at sbt.JLine$$anonfun$withJLine$1.apply(LineReader.scala:115) 
    at sbt.JLine$.withTerminal(LineReader.scala:89) 
    at sbt.JLine$.withJLine(LineReader.scala:115) 
    at sbt.JLine.readLine(LineReader.scala:16) 
    at sbt.BasicCommands$$anonfun$shell$1.apply(BasicCommands.scala:185) 
    at sbt.BasicCommands$$anonfun$shell$1.apply(BasicCommands.scala:181) 
    at sbt.Command$$anonfun$command$1$$anonfun$apply$1.apply(Command.scala:30) 
    at sbt.Command$$anonfun$command$1$$anonfun$apply$1.apply(Command.scala:30) 
    at sbt.Command$.process(Command.scala:93) 
    at sbt.MainLoop$$anonfun$1$$anonfun$apply$1.apply(MainLoop.scala:96) 
    at sbt.MainLoop$$anonfun$1$$anonfun$apply$1.apply(MainLoop.scala:96) 
    at sbt.State$$anon$1.process(State.scala:184) 
    at sbt.MainLoop$$anonfun$1.apply(MainLoop.scala:96) 
    at sbt.MainLoop$$anonfun$1.apply(MainLoop.scala:96) 
    at sbt.ErrorHandling$.wideConvert(ErrorHandling.scala:17) 
    at sbt.MainLoop$.next(MainLoop.scala:96) 
    at sbt.MainLoop$.run(MainLoop.scala:89) 
    at sbt.MainLoop$$anonfun$runWithNewLog$1.apply(MainLoop.scala:68) 
    at sbt.MainLoop$$anonfun$runWithNewLog$1.apply(MainLoop.scala:63) 
    at sbt.Using.apply(Using.scala:24) 
    at sbt.MainLoop$.runWithNewLog(MainLoop.scala:63) 
    at sbt.MainLoop$.runAndClearLast(MainLoop.scala:46) 
    at sbt.MainLoop$.runLoggedLoop(MainLoop.scala:30) 
    at sbt.MainLoop$.runLogged(MainLoop.scala:22) 
    at sbt.StandardMain$.runManaged(Main.scala:57) 
    at sbt.xMain.run(Main.scala:29) 
    at xsbt.boot.Launch$$anonfun$run$1.apply(Launch.scala:57) 
    at xsbt.boot.Launch$.withContextLoader(Launch.scala:77) 
    at xsbt.boot.Launch$.run(Launch.scala:57) 
    at xsbt.boot.Launch$$anonfun$explicit$1.apply(Launch.scala:45) 
    at xsbt.boot.Launch$.launch(Launch.scala:65) 
    at xsbt.boot.Launch$.apply(Launch.scala:16) 
    at xsbt.boot.Boot$.runImpl(Boot.scala:32) 
    at xsbt.boot.Boot$.main(Boot.scala:21) 
    at xsbt.boot.Boot.main(Boot.scala) 
^JException occurred while determining completions. 

この:私は不可解な例外を取得

> tmpTestDynamicParser mo 

> tmpTestDynamicParser move 3 5 

または私は部分入力して、タブを入力すると、 SBTパーサーのための直接的なユースケースでなければならないようですが、私は神秘的な例外を修正する方法については犠牲になり、一般的には動作させます。どんな助けでも大歓迎です。

答えて

0

OK。だから、私はばかだ。

問題はここにあった:

def funcParser(funcs : immutable.Set[Func]) : Parser[Func] = { 
    val funcsByName = funcs.map(func => (func.name, func)).toMap 
    NotSpace.examples(funcsByName.keySet).map(funcsByName) 
} 

NotSpaceその結果がそのMapへのキーとして使用されている場合NoSuchElementExceptionを誘発funcsByNameにキーに含まれていない項目について成功し、有効な結果をもたらすことができます。解決方法は、|と連鎖した一連のリテラルパーサーを使用することでした。

def funcParser(funcs : immutable.Set[Func]) : Parser[Func] = { 
    val funcsByName = funcs.map(func => (func.name, func)).toMap 
    val baseParser = funcsByName.keySet.foldLeft(failure("not a function name") : Parser[String])((nascent, next) => nascent | literal(next)) 
    baseParser.map(processedNamesToFunctions) 
} 
関連する問題