-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathParseState.scala
56 lines (45 loc) · 1.86 KB
/
ParseState.scala
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
package ceedubs.irrec
package regex
import cats.Foldable
import cats.evidence.Is
import cats.implicits._
// TODO consider using something like Stream in some places to get lazy behavior
final case class ParseState[In, A](queue: StateQueue[Thread[In, A]]) extends AnyVal {
import ParseState._
private def threads: List[Thread[In, A]] = queue.reversedElements.reverse
def step(x: In): ParseState[In, A] =
// TODO would it make more sense to use another type of data structure?
threads.foldLeft(ParseState.empty[In, A]) { (st, thread) =>
thread match {
case Thread.Accept(_) => st
case Thread.Cont(_, cont) =>
cont(x).foldLeft(st)(_.addThread(_))
}
}
private def addThread(t: Thread[In, A]): ParseState[In, A] =
t match {
case Thread.Accept(_) => ParseState(queue.insertWithoutId(t))
case Thread.Cont(id, _) => ParseState(queue.insertUnique(id.asInt, t))
}
def results: List[A] = threads.flatMap(_.result)
// TODO document
def parseOnly[F[_]](input: F[In])(implicit foldableF: Foldable[F]): Option[A] =
input
.foldLeftM(this) { (state, input) =>
if (state.queue.reversedElements.isEmpty)
Left(())
else Right(state.step(input))
}
.fold(_ => None, _.results.headOption)
// TODO could add with ops class instead, but run into can't have value class wrapping value class
def parseOnlyS(input: String)(implicit ev: Char Is In): Option[A] =
parseOnly(ev.substitute(input: IndexedSeq[Char]))
}
object ParseState {
def empty[In, A]: ParseState[In, A] = ParseState(StateQueue.empty)
def fromThreads[F[_], In, A](threads: F[Thread[In, A]])(implicit
foldableF: Foldable[F]): ParseState[In, A] =
threads.foldLeft(empty[In, A])(_.addThread(_))
implicit private val indexedSeqFoldable: Foldable[IndexedSeq] =
new IndexedSeqFoldable[IndexedSeq] {}
}