1

使用apollo-server-expressand graphql-tools,我试图从 JSON 对象创建一个最小可行的模式:

const books = [
  {
    "title": "Harry Potter",
    "author": 'J.K. Rowling',
    "slug": "harry_potter",
  },
  {
    "title": 'Jurassic Park',
    "author": 'Michael Crichton',
    "slug": "jurassic_park",
  },
];

// The GraphQL schema in string form
const typeDefs = `
  type Query {
    books: [Book]
    book(title: String!): Book
  }
  type Book { title: String!, author: String!, slug: String! }
`;

// The resolvers
const resolvers = {
  Query: {
    books: () => books,
    book: (_, { title }) => books.filter(book => {
      return new Promise((resolve, reject) => {
        if(book.title == title) {
          console.log('hack log resolve book _: ', JSON.stringify(book))
          resolve(JSON.stringify(book));
        }
      })
    }),
   },
   Book: {
     title: (root, args, context, info) => {
       //args is empty, need to match arg w book.title
       /*
       context:  {
        _extensionStack:
         GraphQLExtensionStack {
           extensions: [ [FormatErrorExtension], [CacheControlExtension] ]
         }
       }
       , root,
       */
       console.log('resolve Book args: ', args, 'info', info);//JSON.stringify(root.book))
       return books.filter(book => {
         if(book.title == root.title) {
           return book;
         }
       });//JSON.stringify({"title": root.title});
     }
   }
};
//    book: (_, { title }) => books.filter(book => book.title == title),


// Put together a schema
const schema = makeExecutableSchema({
  typeDefs,
  resolvers,
});

这是我的存储库

当记录和单步node_modules/graphql/execution/execute.js执行时,execute 的第一个参数argsOrSchema.variableValues包含查询参数键和值,但是第五个参数variableValues是未定义的。

根据诸如this GitHub issue之类的一些线程,我可以variableValuesinfo解析器的参数中提取,但是我仍然想知道为什么该args对象为空?

这是GraphQL 在解析器函数中给出的信息日志的要点

4

1 回答 1

3

参数由传递给正在解析的字段的args参数填充——传递给其他字段的任何参数都不会包含在args参数中。

您的架构在您的类型的字段上包含一个参数 ( title) 。这意味着该字段的解析器将接收参数作为其参数的一部分,但前提是该参数实际包含在您的查询中:bookQuerytitleargs

// Request
query {
  book(title: "Something") {
    title
  }
}

// Resolvers
const resolvers = {
  Query: {
    book: (root, args) => {
      console.log(args) // {title: 'Something'}
    }
  },
}

相对于:

// Request
query {
  book {
    title
  }
}

// Resolvers
const resolvers = {
  Query: {
    book: (root, args) => {
      console.log(args) // {}
    }
  },
}

如果您为title参数传入一个值,那么在其他字段的解析器中获取该值的唯一方法是解析info参数。但是,您不会查看variableValues属性,因为传递给参数的值可能是文字值或变量。您需要遍历fieldNodes数组并找到适当的参数值。

但是,通常不需要经历所有这些。

如果该book字段应该只返回一个书籍对象,则从数组中选择正确书籍的逻辑books应该包含在该字段的解析器中:

const resolvers = {
  Query: {
    book: (root, args) => {
      return books.find(book => book.title === args.title)
    }
  },
}

没有理由title在类型上包含字段的解析器Book,除非您需要该字段解析为默认情况下将解析的其他内容(title父字段的解析器返回的对象的属性)。这足以按标题查询所有书籍和单个书籍:

const resolvers = {
  Query: {
    book: (root, args) => {
      return books.find(book => book.title === args.title)
    },
    books: () => books,
  },
}

查看 Apollo 的官方教程以获取更多示例以及解析器如何工作的完整说明。

于 2019-04-16T21:04:43.383 回答