缺少对象 GraphQL+Apollo 的选择集错误

IT技术 javascript reactjs graphql apollo react-apollo
2021-05-10 23:05:36

我有一组触发某些类型弹出窗口的本地状态的突变。他们一般是这样设置的:

  openDialog: (_, variables, { cache }) => {
    const data = {
      popups: {
        ...popups,
        dialog: {
          id: 'dialog',
          __typename: 'Dialog',
          type: variables.type
        }
      }
    };

    cache.writeData({
      data: data
    });
    return null;
  }

我传入的默认值如下所示:

const defaults = {
  popups: {
    __typename: TYPENAMES.POPUPS,
    id,
    message: null,
    modal: null,
    menu: null,
    dialog: null
  }
};

它们在我的 React 代码中的使用方式是使用 Mutation 包装器组件,如下所示:

const OPEN_ALERT_FORM = gql`
  mutation AlertOpenDialog($type: String!) {
    openDialog(type: $type) @client
  }
`;

class Alert extends Component {
  render() {
    return (
      <Mutation mutation={OPEN_ALERT_FORM} variables={{ type: ALERT_FORM }}>
        {openDialog => {
          return (
            <Button
              classes="alert-button"
              onClick={openDialog}
              label="Trigger Alert"
            />
          );
        }}
      </Mutation>
    );
  }
}

对于我的各种弹出窗口(我有 3 或 4 个不同的弹出窗口,例如menumodal),打开和关闭它们的更改看起来都一样,只是类型名称和内容不同等。但是,对于对话框,当我单击它们时出现此错误:

网络错误:缺少为查询字段对话框返回的 Dialog 类型对象的选择集

...然后触发组件从页面上消失。另外,一旦发生这种情况,当您尝试单击它们时,所有其他弹出类型都会消失,然后重新抛出该错误,或​​者说:

未捕获的错误:引发了跨源错误。React 无权访问开发中的实际错误对象。

我尝试重新编写对话框以匹配其他弹出类型,并重新编写组件,但我仍然收到此错误。它似乎是特定于对话+阿波罗的。这个问题的根源是什么?它不能是后端的东西,因为这只是与本地 Apollo 打交道。我以前没有见过这个错误,我不知道从哪里开始。

3个回答

解决方案是向查询添加字段(与声明要获取的顶级对象而不指定要获取的字段相比)。如果你有类似的东西:

{
  popups @client {
    id
    dialog
  }
}

您必须声明一些字段以在内部获取dialog,例如id

{
  popups @client {
    id
    dialog {
      id
    }
  }
}

实际问题的答案似乎在查询中。最初 Apollo 客户端没有验证@client查询/变异的类型,所以你的变异看起来就像你在问题中写的一样:

mutation AlertOpenDialog($type: String!) {
  openDialog(type: $type) @client
}

上面的正确书写方式是指定(选择)您想要在响应中获得的所有简单类型(标量)字段。所以关于@Vic 的回答,突变应该更像:

mutation AlertOpenDialog($type: String!) {
  openDialog(type: $type) @client {
    dialog {
      id
    }
  }
}

解决这个问题的方法是将dialog字段视为字符串而不是对象。将函数更改为此可以解决问题并使错误消失:

  openDialog: (_, variables, { cache }) => {
    const data = {
      popups: {
        ...popups,
        dialog: variables.type
      }
    };

    cache.writeData({
      data: data
    });
    return null;
  }