From c86d2aabc4825f3a2c9a345dc3834aabfaea273b Mon Sep 17 00:00:00 2001 From: Kyri Petrou Date: Thu, 23 May 2024 17:21:28 +0200 Subject: [PATCH 1/5] Parser micro-optimizations --- .../scala/caliban/ComplexQueryBenchmark.scala | 183 +++++++++--------- .../main/scala/caliban/ParserBenchmark.scala | 30 ++- .../caliban/parsing/parsers/Parsers.scala | 5 +- .../parsing/parsers/SelectionParsers.scala | 13 +- .../parsing/parsers/StringParsers.scala | 2 +- .../parsing/parsers/ValueParsers.scala | 3 - 6 files changed, 105 insertions(+), 131 deletions(-) diff --git a/benchmarks/src/main/scala/caliban/ComplexQueryBenchmark.scala b/benchmarks/src/main/scala/caliban/ComplexQueryBenchmark.scala index bc69e2dac6..fa260c2c95 100644 --- a/benchmarks/src/main/scala/caliban/ComplexQueryBenchmark.scala +++ b/benchmarks/src/main/scala/caliban/ComplexQueryBenchmark.scala @@ -58,97 +58,94 @@ class ComplexQueryBenchmark { } object ComplexQueryBenchmark { - val fullIntrospectionQuery = """ - query IntrospectionQuery { - __schema { - queryType { name } - mutationType { name } - subscriptionType { name } - types { - ...FullType - } - directives { - name - description - locations - args { - ...InputValue - } - } - } - } - - fragment FullType on __Type { - kind - name - description - fields(includeDeprecated: true) { - name - description - args { - ...InputValue - } - type { - ...TypeRef - } - isDeprecated - deprecationReason - } - inputFields { - ...InputValue - } - interfaces { - ...TypeRef - } - enumValues(includeDeprecated: true) { - name - description - isDeprecated - deprecationReason - } - possibleTypes { - ...TypeRef - } - } - - fragment InputValue on __InputValue { - name - description - type { ...TypeRef } - defaultValue - } - - fragment TypeRef on __Type { - kind - name - ofType { - kind - name - ofType { - kind - name - ofType { - kind - name - ofType { - kind - name - ofType { - kind - name - ofType { - kind - name - ofType { - kind - name - } - } - } - } - } - } - } - } - """ + val fullIntrospectionQuery = + """query IntrospectionQuery { + | __schema { + | queryType { name } + | mutationType { name } + | subscriptionType { name } + | types { + | ...FullType + | } + | directives { + | name + | description + | locations + | args { + | ...InputValue + | } + | } + | } + |} + |fragment FullType on __Type { + | kind + | name + | description + | fields(includeDeprecated: true) { + | name + | description + | args { + | ...InputValue + | } + | type { + | ...TypeRef + | } + | isDeprecated + | deprecationReason + | } + | inputFields { + | ...InputValue + | } + | interfaces { + | ...TypeRef + | } + | enumValues(includeDeprecated: true) { + | name + | description + | isDeprecated + | deprecationReason + | } + | possibleTypes { + | ...TypeRef + | } + |} + |fragment InputValue on __InputValue { + | name + | description + | type { ...TypeRef } + | defaultValue + |} + |fragment TypeRef on __Type { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | ofType { + | kind + | name + | } + | } + | } + | } + | } + | } + | } + |} + |""".stripMargin } diff --git a/benchmarks/src/main/scala/caliban/ParserBenchmark.scala b/benchmarks/src/main/scala/caliban/ParserBenchmark.scala index e1178714fa..b72e1e2411 100644 --- a/benchmarks/src/main/scala/caliban/ParserBenchmark.scala +++ b/benchmarks/src/main/scala/caliban/ParserBenchmark.scala @@ -1,22 +1,18 @@ package caliban import caliban.parsing.Parser -import cats.effect.IO -import io.circe.Json import org.openjdk.jmh.annotations._ -import sangria.execution._ -import sangria.marshalling.circe._ +import org.openjdk.jmh.infra.Blackhole import sangria.parser.QueryParser import java.util.concurrent.TimeUnit -import scala.concurrent.duration._ -import scala.concurrent.{ Await, ExecutionContextExecutor, Future } +import scala.concurrent.ExecutionContextExecutor @State(Scope.Thread) @BenchmarkMode(Array(Mode.Throughput)) @OutputTimeUnit(TimeUnit.SECONDS) -@Warmup(iterations = 5, time = 3, timeUnit = TimeUnit.SECONDS) -@Measurement(iterations = 5, time = 3, timeUnit = TimeUnit.SECONDS) +@Warmup(iterations = 5, time = 1, timeUnit = TimeUnit.SECONDS) +@Measurement(iterations = 5, time = 1, timeUnit = TimeUnit.SECONDS) @Fork(1) class ParserBenchmark { import ComplexQueryBenchmark._ @@ -24,28 +20,26 @@ class ParserBenchmark { implicit val executionContext: ExecutionContextExecutor = scala.concurrent.ExecutionContext.global @Benchmark - def runCaliban(): Unit = { - val io = Parser.parseQuery(fullIntrospectionQuery) - Caliban.run(io) + def runCaliban(bh: Blackhole): Unit = { + bh.consume(Parser.parseQueryEither(fullIntrospectionQuery).fold(throw _, identity)) () } @Benchmark - def runSangria(): Unit = { - val future = Future.fromTry(QueryParser.parse(fullIntrospectionQuery)) - Await.result(future, 1.minute) + def runSangria(bh: Blackhole): Unit = { + bh.consume(QueryParser.parse(fullIntrospectionQuery).fold(throw _, identity)) () } @Benchmark - def runGrackle(): Unit = { - Grackle.compiler.compile(fullIntrospectionQuery) + def runGrackle(bh: Blackhole): Unit = { + bh.consume(Grackle.compiler.compile(fullIntrospectionQuery)) () } @Benchmark - def runGql(): Unit = { - gql.parser.parseQuery(fullIntrospectionQuery) + def runGql(bh: Blackhole): Unit = { + bh.consume(gql.parser.parseQuery(fullIntrospectionQuery)) () } } diff --git a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala index 6d26f02dd7..f4279f5b81 100644 --- a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala @@ -13,9 +13,6 @@ import caliban.parsing.adt.Type._ import caliban.parsing.adt._ import fastparse._ -import scala.annotation.nowarn - -@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.x object Parsers extends SelectionParsers { def argumentDefinition(implicit ev: P[Any]): P[InputValueDefinition] = (stringValue.? ~ name ~ ":" ~ type_ ~ defaultValue.? ~ directives.?).map { @@ -348,7 +345,7 @@ object Parsers extends SelectionParsers { schemaExtension | typeExtension def definition(implicit ev: P[Any]): P[Definition] = - executableDefinition | typeSystemDefinition | typeSystemExtension + typeSystemDefinition | typeSystemExtension def document(implicit ev: P[Any]): P[ParsedDocument] = ((Start ~ executableDefinition.rep ~ End) | (Start ~ definition.rep ~ End)).map(seq => ParsedDocument(seq.toList)) diff --git a/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala b/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala index 33c1fbe3f8..29004ad7e2 100644 --- a/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala @@ -6,13 +6,8 @@ import caliban.parsing.adt.Type._ import caliban.parsing.adt._ import fastparse._ -import scala.annotation.nowarn - -@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.x private[caliban] trait SelectionParsers extends ValueParsers { - @deprecated("Kept for bincompat only, scheduled to be removed") - def alias(implicit ev: P[Any]): P[String] = name ~ ":" def aliasOrName(implicit ev: P[Any]): P[String] = ":" ~/ name def argument(implicit ev: P[Any]): P[(String, InputValue)] = name ~ ":" ~ value @@ -30,19 +25,13 @@ private[caliban] trait SelectionParsers extends ValueParsers { def namedType(implicit ev: P[Any]): P[NamedType] = name.filter(_ != "null").map(NamedType(_, nonNull = false)) def listType(implicit ev: P[Any]): P[ListType] = ("[" ~ type_ ~ "]").map(t => ListType(t, nonNull = false)) - @deprecated("Kept for bincompat only, scheduled to be removed") - def nonNullType(implicit ev: P[Any]): P[Type] = ((namedType | listType) ~ "!").map { - case t: NamedType => t.copy(nonNull = true) - case t: ListType => t.copy(nonNull = true) - } - def type_(implicit ev: P[Any]): P[Type] = ((namedType | listType) ~ "!".!.?).map { case (t: NamedType, nn) => if (nn.isDefined) t.copy(nonNull = true) else t case (t: ListType, nn) => if (nn.isDefined) t.copy(nonNull = true) else t } def field(implicit ev: P[Any]): P[Field] = - (Index ~ name ~ aliasOrName.? ~ arguments.? ~ directives.? ~ selectionSet.?).map { + (Index ~~ name ~ aliasOrName.? ~ arguments.? ~ directives.? ~ selectionSet.?).map { case (index, alias, Some(name), args, dirs, sels) => Field(Some(alias), name, args.getOrElse(Map()), dirs.getOrElse(Nil), sels.getOrElse(Nil), index) case (index, name, _, args, dirs, sels) => diff --git a/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala b/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala index 384e4511b9..71a55536c4 100644 --- a/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala @@ -57,7 +57,7 @@ private[caliban] trait StringParsers { def sourceCharacter(implicit ev: P[Any]): P[Unit] = CharIn("\u0009\u000A\u000D\u0020-\uFFFF") def sourceCharacterWithoutLineTerminator(implicit ev: P[Any]): P[Unit] = CharIn("\u0009\u0020-\uFFFF") - def name(implicit ev: P[Any]): P[String] = (CharIn("_A-Za-z") ~~ CharIn("_0-9A-Za-z").repX).! + def name(implicit ev: P[Any]): P[String] = !CharIn("0-9") ~~ CharsWhileIn("_0-9A-Za-z", 1).! def nameOnly(implicit ev: P[Any]): P[String] = Start ~ name ~ End def hexDigit(implicit ev: P[Any]): P[Unit] = CharIn("0-9a-fA-F") diff --git a/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala b/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala index 63182b3c6b..1bf330664c 100644 --- a/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala @@ -5,9 +5,6 @@ import caliban.InputValue._ import caliban.Value._ import fastparse._ -import scala.annotation.nowarn - -@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.x private[caliban] trait ValueParsers extends NumberParsers { def booleanValue(implicit ev: P[Any]): P[BooleanValue] = StringIn("true", "false").!.map(v => BooleanValue(v.toBoolean)) From 2d24ffb6e7de67ef6a69dc4f9e5342df4bfe9b37 Mon Sep 17 00:00:00 2001 From: Kyri Petrou Date: Thu, 23 May 2024 17:32:11 +0200 Subject: [PATCH 2/5] Use flatMap instead of positive-lookahead --- .../main/scala/caliban/parsing/parsers/StringParsers.scala | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala b/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala index 71a55536c4..59e9678dac 100644 --- a/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/StringParsers.scala @@ -57,7 +57,12 @@ private[caliban] trait StringParsers { def sourceCharacter(implicit ev: P[Any]): P[Unit] = CharIn("\u0009\u000A\u000D\u0020-\uFFFF") def sourceCharacterWithoutLineTerminator(implicit ev: P[Any]): P[Unit] = CharIn("\u0009\u0020-\uFFFF") - def name(implicit ev: P[Any]): P[String] = !CharIn("0-9") ~~ CharsWhileIn("_0-9A-Za-z", 1).! + def name(implicit ev: P[Any]): P[String] = + CharsWhileIn("_0-9A-Za-z", 1).!.flatMap { s => + // Less efficient in case of an error, but more efficient in case of success (happy path) + if (s.charAt(0) <= '9') ev.freshFailure() + else ev.freshSuccess(s) + } def nameOnly(implicit ev: P[Any]): P[String] = Start ~ name ~ End def hexDigit(implicit ev: P[Any]): P[Unit] = CharIn("0-9a-fA-F") From 9a0bea56285ba6fc52902e293f166cc612405df9 Mon Sep 17 00:00:00 2001 From: Kyri Petrou Date: Thu, 23 May 2024 17:36:36 +0200 Subject: [PATCH 3/5] Add back nowarn annotations --- core/src/main/scala/caliban/parsing/parsers/Parsers.scala | 3 +++ .../main/scala/caliban/parsing/parsers/SelectionParsers.scala | 3 +++ core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala | 3 +++ 3 files changed, 9 insertions(+) diff --git a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala index f4279f5b81..d10f3b3235 100644 --- a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala @@ -13,6 +13,9 @@ import caliban.parsing.adt.Type._ import caliban.parsing.adt._ import fastparse._ +import scala.annotation.nowarn + +@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.12.x object Parsers extends SelectionParsers { def argumentDefinition(implicit ev: P[Any]): P[InputValueDefinition] = (stringValue.? ~ name ~ ":" ~ type_ ~ defaultValue.? ~ directives.?).map { diff --git a/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala b/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala index 29004ad7e2..9b4d2e2eef 100644 --- a/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/SelectionParsers.scala @@ -6,6 +6,9 @@ import caliban.parsing.adt.Type._ import caliban.parsing.adt._ import fastparse._ +import scala.annotation.nowarn + +@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.12.x private[caliban] trait SelectionParsers extends ValueParsers { def aliasOrName(implicit ev: P[Any]): P[String] = ":" ~/ name diff --git a/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala b/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala index 1bf330664c..724eb708c4 100644 --- a/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/ValueParsers.scala @@ -5,6 +5,9 @@ import caliban.InputValue._ import caliban.Value._ import fastparse._ +import scala.annotation.nowarn + +@nowarn("msg=NoWhitespace") // False positive warning in Scala 2.12.x private[caliban] trait ValueParsers extends NumberParsers { def booleanValue(implicit ev: P[Any]): P[BooleanValue] = StringIn("true", "false").!.map(v => BooleanValue(v.toBoolean)) From 33a65120814e21c351513f47afaf7631c738ff0b Mon Sep 17 00:00:00 2001 From: Kyri Petrou Date: Fri, 24 May 2024 10:36:21 +0200 Subject: [PATCH 4/5] Revert change to `definition` --- core/src/main/scala/caliban/parsing/parsers/Parsers.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala index d10f3b3235..0ca5a111f3 100644 --- a/core/src/main/scala/caliban/parsing/parsers/Parsers.scala +++ b/core/src/main/scala/caliban/parsing/parsers/Parsers.scala @@ -348,7 +348,7 @@ object Parsers extends SelectionParsers { schemaExtension | typeExtension def definition(implicit ev: P[Any]): P[Definition] = - typeSystemDefinition | typeSystemExtension + executableDefinition | typeSystemDefinition | typeSystemExtension def document(implicit ev: P[Any]): P[ParsedDocument] = ((Start ~ executableDefinition.rep ~ End) | (Start ~ definition.rep ~ End)).map(seq => ParsedDocument(seq.toList)) From fbc35366beb687eec8a8e46025a5b9a84d4812ce Mon Sep 17 00:00:00 2001 From: Kyri Petrou Date: Fri, 24 May 2024 10:53:16 +0200 Subject: [PATCH 5/5] Remove blackhole usage from parser benchmarks --- .../main/scala/caliban/ParserBenchmark.scala | 30 ++++++++----------- 1 file changed, 13 insertions(+), 17 deletions(-) diff --git a/benchmarks/src/main/scala/caliban/ParserBenchmark.scala b/benchmarks/src/main/scala/caliban/ParserBenchmark.scala index b72e1e2411..a20f5306b0 100644 --- a/benchmarks/src/main/scala/caliban/ParserBenchmark.scala +++ b/benchmarks/src/main/scala/caliban/ParserBenchmark.scala @@ -1,8 +1,12 @@ package caliban import caliban.parsing.Parser +import caliban.parsing.adt.Document +import cats.data.NonEmptyList +import cats.parse.Caret +import gql.parser.QueryAst +import grackle.Operation import org.openjdk.jmh.annotations._ -import org.openjdk.jmh.infra.Blackhole import sangria.parser.QueryParser import java.util.concurrent.TimeUnit @@ -20,26 +24,18 @@ class ParserBenchmark { implicit val executionContext: ExecutionContextExecutor = scala.concurrent.ExecutionContext.global @Benchmark - def runCaliban(bh: Blackhole): Unit = { - bh.consume(Parser.parseQueryEither(fullIntrospectionQuery).fold(throw _, identity)) - () - } + def runCaliban(): Document = + Parser.parseQueryEither(fullIntrospectionQuery).fold(throw _, identity) @Benchmark - def runSangria(bh: Blackhole): Unit = { - bh.consume(QueryParser.parse(fullIntrospectionQuery).fold(throw _, identity)) - () - } + def runSangria(): sangria.ast.Document = + QueryParser.parse(fullIntrospectionQuery).fold(throw _, identity) @Benchmark - def runGrackle(bh: Blackhole): Unit = { - bh.consume(Grackle.compiler.compile(fullIntrospectionQuery)) - () - } + def runGrackle(): Operation = + Grackle.compiler.compile(fullIntrospectionQuery).getOrElse(throw new Throwable("Grackle failed to parse query")) @Benchmark - def runGql(bh: Blackhole): Unit = { - bh.consume(gql.parser.parseQuery(fullIntrospectionQuery)) - () - } + def runGql(): NonEmptyList[QueryAst.ExecutableDefinition[Caret]] = + gql.parser.parseQuery(fullIntrospectionQuery).fold(e => throw new Throwable(e.prettyError.value), identity) }