graphql - 在 GraphQL 中使用查询结果作为下一级别的参数

标签 graphql apollo

大家好,

之前已经讨论过这一点,但这是其中之一,有如此多的分散讨论导致各种提议的“黑客”,我很难确定我应该做什么。

我想使用查询的结果作为另一个嵌套查询的参数。

query {
  allStudents {
    nodes {
      courseAssessmentInfoByCourse(courseId: "2b0df865-d7c6-4c96-9f10-992cd409dedb") {
        weightedMarkAverage
        // getting result for specific course is easy enough
      }
      coursesByStudentCourseStudentIdAndCourseId {
        nodes {
          name
          // would like to be able to do something like this
          // to get a list of all the courses and their respective
          // assessment infos
          assessmentInfoByStudentId (studentId: student_node.studentId) {
            weightedMarkAverage
          }
        }
      }
    }
  }
}

有没有一种被认为是最佳实践的方法?
现在是否有一种标准的方法可以内置到 GraphQL 中?

谢谢你的帮助!

最佳答案

替换 GraphQL 文档中的值的唯一方法是通过变量,这些必须在您的操作定义中声明,然后作为您请求的一部分包含在您的文档中。 没有固有的方法来引用同一文档中先前解析的值。

如果您认为您需要此功能,那通常首先是模式设计不佳的症状。下面是一些改进架构的建议,假设您可以控制它。

例如,您至少可以消除 studentId关于 assessmentInfoByStudentId 的争论共。 coursesByStudentCourseStudentIdAndCourseId是学生节点上的一个字段,所以它的解析器已经可以访问学生的 id。它可以将此信息向下传递到每个类(class)节点,然后可以由 assessmentInfoByStudentId 使用。 .

也就是说,您最好完全重新考虑如何建立连接。我不知道你的底层存储层是什么样的,或者你的客户需要数据的形状,所以很难提出任何具体的建议。但是,为了举例,我们假设我们有三种类型——Course , StudentAssessmentInfo .一个 Course有很多Students , Student有很多Courses , 和 AssessmentInfo有一个Student和一个Course .

我们可能会将所有三个实体公开为根级查询:

query {
  allStudents {
    # fields
  }
  allCourses {
    # fields
  }
  allAssessmentInfos {
    # fields
  }
}

每个节点都可以连接到其他两种类型:
query {
  allStudents {
    courses {
      edges {
        node {
          id
        }
      }
    }
    assessmentInfos {
      edges {
        node {
          id
        }
      }
    }

  }
}

如果我们想获取所有学生,并且每个学生都知道他/她正在学习哪些类(class)以及他/她对该类(class)的加权平均分,我们可以编写如下查询:
query {
  allStudents {
    assessmentInfos {
      edges {
        node {
          id
          course {
            id
            name
          }
        }
      }
    }

  }
}

同样,这个确切的模式可能不适用于您的特定用例,但它应该让您了解如何从不同的角度解决您的问题。设计模式时的更多提示:
  • 在连接字段上添加过滤器参数,而不是为您需要涵盖的每个场景创建单独的字段。单courses字段 Student type 可以有多种参数,例如 semester , campusisPassing -- 这比创建不同的字段(如 coursesBySemester)更简洁、更灵活, coursesByCampus
  • 如果您正在处理诸如平均值、最小值、最大值等聚合值。将这些值公开为每个连接类型的字段可能是有意义的,就像 count 一样。字段有时与 nodes 一起提供 field 。有一个(提案)[ https://github.com/prisma/prisma/issues/1312]对于 Prisma,它说明了处理这些聚合值的一种相当巧妙的方法。例如,如果您已经拥有一个 Assessment,那么做这样的事情就意味着类型,连接字段可能足以公开有关该类型的聚合数据(如平均成绩),而无需公开单独的 AssessmentInfo类型。
  • 过滤相对简单,分组有点困难。如果您确实发现需要按特定字段分组的连接节点,最好再次通过在连接本身上公开一个附加字段来完成此操作(就像 Gatsby 所做的那样)[ https://www.gatsbyjs.org/docs/graphql-reference/#group] .
  • 关于graphql - 在 GraphQL 中使用查询结果作为下一级别的参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54192685/

    相关文章:

    express - 将获取响应 header 转发到 Apollo graphql 响应

    reactjs - Graphql:如果参数类型为字符串,则跳过该参数

    ios - 安装 GraphQL iOS 框架

    graphql - 测试运行完成后,Jest 一秒钟都没有退出。 --检测打开句柄

    javascript - graphql 突变中无法识别的参数

    c# - Apollo Stomp ActiveMQ 创建无效的临时目的地名称

    graphql - 查询graphiql导致Apollo错误forward is not a function

    react-native - 如何在react-apollo graphql查询中正确使用动态变量?

    javascript - Angular - Apollo : Client has not been defined yet

    node.js - 如何在 NodeJS 中使用 GraphQL 和 Apollo 上传服务器上传文件?