在GraphQL的下一级别中将查询结果用作参数

时间:2019-01-15 04:25:28

标签: 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中吗?

感谢您的帮助!

1 个答案:

答案 0 :(得分:1)

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

如果到达需要使用此功能的地步,通常首先是架构设计不佳的征兆。假设您对此有控制权,那么以下是一些改进架构的建议。

例如,至少可以完全消除studentId上的assessmentInfoByStudentId参数。 coursesByStudentCourseStudentIdAndCourseId是学生节点上的字段,因此其解析程序已经可以访问学生的ID。它可以将此信息向下传递到每个课程节点,然后由assessmentInfoByStudentId使用。

也就是说,您最好完全重新考虑如何建立连接。我不知道您的基础存储层是什么样子,也不知道您的客户端需要数据的形状,因此很难提出任何具体建议。但是,为便于举例,我们假设我们有三种类型-CourseStudentAssessmentInfoCourse有许多StudentsStudent有许多Courses,而AssessmentInfo有一个Student和一个{{1} }。

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

Course

每个节点都可以与其他两种类型建立连接:

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

如果我们想获取所有学生,并且让每个学生知道他/他正在修什么课程以及他/她对该课程的加权平均分,那么我们可以编写如下查询:

query {
  allStudents {
    courses {
      edges {
        node {
          id
        }
      }
    }
    assessmentInfos {
      edges {
        node {
          id
        }
      }
    }

  }
}

同样,这种确切的模式可能不适用于您的特定用例,但它应该使您了解如何从另一个角度解决问题。设计架构时,还有另外两个技巧:

  • 在连接字段上添加过滤器参数,而不是为需要涵盖的每种情况创建单独的字段。 query { allStudents { assessmentInfos { edges { node { id course { id name } } } } } } 类型上的单个courses字段可以具有各种参数,例如Studentsemestercampus -比创建更干净,更灵活isPassingcoursesBySemester等不同的字段。
  • 如果要处理诸如平均值,最小值,最大值之类的合计值,那么将这些值公开为每种连接类型上的字段可能是有意义的,就像有时使用coursesByCampus字段一样count字段。 Prisma有一个(建议)[https://github.com/prisma/prisma/issues/1312],它说明了一种处理这些汇总值的相当简洁的方法。进行类似的操作将意味着,如果您已经拥有nodes类型,则连接字段可能足以暴露有关该类型的汇总数据(例如平均成绩),而无需暴露单独的{{1} }类型。
  • 过滤相对简单,而分组则比较困难。如果确实发现需要按特定字段分组的连接节点,则最好再次在连接本身上暴露一个附加字段来完成此操作(就像Gatsby一样)[https://www.gatsbyjs.org/docs/graphql-reference/#group]