All Downloads are FREE. Search and download functionalities are using the official Maven repository.

gql.parser.QueryParser.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2024 Valdemar Grange
 *
 * 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 gql.parser

import cats.parse.{Parser => P}
import cats.parse.Caret

// https://spec.graphql.org/June2018/#sec-Source-Text
object QueryParser {
  import QueryAst._
  import GraphqlParser._

  lazy val executableDefinition = {
    import ExecutableDefinition._
    Pos.pos(fragmentDefinition).map(p => Fragment(p.value, p.caret)) |
      Pos.pos(operationDefinition).map(p => Operation(p.value, p.caret))
  }

  lazy val operationDefinition: P[OperationDefinition[Caret]] = {
    import OperationDefinition._
    P.backtrack(selectionSet).map(Simple(_)) |
      (operationType ~ name.? ~ variableDefinitions.? ~ directivesAny.? ~ selectionSet).map { case ((((opt, name), vars), ds), ss) =>
        Detailed(opt, name, vars, ds, ss)
      }
  }

  lazy val variableDefinitions =
    variableDefinition.rep
      .between(t('('), t(')'))
      .map(VariableDefinitions(_))

  lazy val variableDefinition =
    Pos.pos(variable ~ (t(':') *> `type`) ~ defaultValue(constValue).? ~ directivesConst.?).map { case Pos(c, (((n, t), d), ds)) =>
      VariableDefinition(n, t, d, ds, c)
    }

  lazy val operationType = {
    import OperationType._

    s("query").as(Query) |
      s("mutation").as(Mutation) |
      s("subscription").as(Subscription)
  }

  lazy val selectionSet: P[SelectionSet[Caret]] = P.defer {
    selection.rep.between(t('{'), t('}')).map(SelectionSet(_))
  }

  lazy val selection: P[Selection[Caret]] = {
    import Selection._
    Pos.pos(field).map(p => FieldSelection(p.value, p.caret)) |
      // expects on, backtrack on failure
      Pos.pos(inlineFragment).map(p => InlineFragmentSelection(p.value, p.caret)) |
      Pos.pos(fragmentSpread).map(p => FragmentSpreadSelection(p.value, p.caret))
  }

  lazy val field: P[Field[Caret]] = P.defer {
    Pos
      .pos(P.backtrack(alias).?.with1 ~ name ~ argumentsAny.? ~ directivesAny.? ~ selectionSet.?)
      .map { case Pos(c, ((((a, n), args), ds), s)) => Field(a, n, args, ds, s, c) }
  }

  lazy val alias = name <* t(':')

  lazy val fragmentSpread =
    (s("...") *> fragmentName ~ directivesAny.?).map { case (n, ds) => FragmentSpread.apply(n, ds) }

  lazy val inlineFragment =
    ((s("...") *> typeCondition.? ~ directivesAny.?).soft ~ selectionSet).map { case ((t, ds), s) => InlineFragment(t, ds, s) }

  lazy val fragmentDefinition =
    Pos.pos(s("fragment") *> fragmentName ~ typeCondition ~ directivesAny.? ~ selectionSet).map { case Pos(c, (((n, t), ds), s)) =>
      FragmentDefinition(n, t, ds, s, c)
    }

  lazy val fragmentName: P[String] =
    (!s("on")).with1 *> name

  lazy val typeCondition: P[String] =
    s("on") *> name
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy