【问题标题】:How can I handle input errors in HotChocolate?如何处理 HotChocolate 中的输入错误?
【发布时间】:2021-07-13 14:01:10
【问题描述】:

我有由 HotChocolate 集成的带有 Graphql 的 Asp.Net Core Api。我有输入类型 CreateRoadmapInput 的变异 CreateRoadmap。 My MutationMy input type.


我需要正确验证这些输入的字段。我尝试使用 IErrorFilter.My Error Filter 我设法自定义错误消息,但它总是返回 500 状态代码并且没有指定具有无效输入值的字段。例如:客户端可以将 GraphQl 变量的一个字段作为浮点数发送(后面带有点和数字)。在这种情况下,我收到了状态代码为 500 的内部服务器错误和错误消息。 Response with invalid inputResponse Body。俄语翻译:“输入数据无效:字段值无效:请输入整数”


我想要什么:

  1. 在输入无效的情况下给出 400(Bad Request) 状态码。
  2. 指定哪些字段的值无效

对不起,我的英语不好。希望你明白这个问题。感谢您的帮助!


我的突变

public ObjectPayload<Roadmap> CreateRoadmap(
        CreateRoadmapInput obj,
        [Service] IServiceFactory<Roadmap> serviceFactory)
    {
        try
        {
            var roadmapService = serviceFactory.CreateServiceInstance(obj.WorkerType);
            var roadmap = roadmapService.Create(obj);

            return new ObjectPayload<Roadmap>(roadmap);
        }
        catch (Exception e)
        {
            return new ObjectPayload<Roadmap>(
                new UserError(e.Message, "EXECUTION_ERROR"));
        }
    }

Mu 输入类型

public class CreateRoadmapInput : InputBase
{
    
    public int WorkerId { get; set; }
    public RoadmapType Type { get; set; }
    public int Period { get; set; }
    public int Year { get; set; }
    public int ProcessPercent { get; set; }
    public int ProjectPercent { get; set; }
    public WorkerType WorkerType { get; set; }
}

我的错误过滤器

public class ErrorHandling : IErrorFilter
{
    public IError OnError(IError error)
    {
        error = error.RemoveExtensions().RemoveLocations().RemoveCode();
        if (error.Message.Contains("got an invalid value"))
        {
            if (error.Exception.Message.Contains("Int cannot parse the given literal of type"))
                return error.WithMessage($"Некорректные вводимые данные: Неверное значение поля. Введите целое число");
            else
                return error.WithMessage($"Некорректные вводимые данные: {error.Exception.Message}");
        }
        return error.WithMessage(error.Exception.Message);
    }
}

【问题讨论】:

  • 您好,欢迎来到 SO。感谢您发布您的问题。您可以复制问题中的代码而不是图像吗?根据stackoverflow.com/help/how-to-ask:“请勿发布代码、数据、错误消息等的图像”。感谢您的理解
  • 好的,我一拿到电脑就复制代码

标签: c# asp.net-core graphql hotchocolate


【解决方案1】:

graphql 处理管道分为验证和执行。据我了解,Sergey Shaykhullin 提出的用于输入验证的自定义库仅在执行阶段才开始参与该过程。 但是您提到的变量的情况是a part of 验证阶段。 实际上,版本 11.2.2 验证引擎的 HotChocolate 为您提供了您需要的所有数据(即特定的错误代码和变量名称) “扩展”字段:

{
    "errors": [
        {
            "message": "Variable `someInt` got an invalid value.",
            "locations": [
                {
                    "line": 1,
                    "column": 21
                }
            ],
            "extensions": {
                "code": "HC0016",
                "variable": "someInt",
                "underlyingError": "Int cannot parse the given literal of type `StringValueNode`."
            }
        }
    ]
}

关于状态码 400 的第二个问题,是的,错误 500 在这里真的很奇怪。我相信验证错误的错误 500 不符合 graphql over http 规范。该规范明确指出

“服务器应该拒绝执行,状态码为 400”

要解决这个问题,您可以写your own result serializer。在最简单的形式中,它可能类似于以下形式:

public class HttpResultSerializerWithCustomStatusCodes : DefaultHttpResultSerializer
    {
        public override HttpStatusCode GetStatusCode(IExecutionResult result)
        {
            var baseStatusCode = base.GetStatusCode(result);

            if (result is IQueryResult && baseStatusCode == HttpStatusCode.InternalServerError && result.Errors?.Count > 0)
            {
                if (result.Errors.Any(e => e.Code == ErrorCodes.Authentication.NotAuthorized || e.Code == ErrorCodes.Authentication.NotAuthenticated))
                    return HttpStatusCode.Forbidden;    

                return HttpStatusCode.BadRequest;    

            }

            return baseStatusCode;
        }
    }

【讨论】:

  • 感谢您的出色回答。我将尝试编写自己的结果序列化程序以返回 400 状态代码,以防 GraphQL var 出现问题。没有办法处理 500 个代码状态 - 前线会活活吃掉我。我会尽快在这里写下结果
  • 是的,客户端框架上的代码可能存在问题,例如,在 apollo 客户端上。顺便说一句,您使用的哪个客户端框架(如果有)面临 graphql 服务器状态 500 返回的问题?
  • 我们将 Apollo 客户端用于 graphQl。
  • 和我们一样。伙计们,你们遇到有关来自 graphql 服务器的错误 4xx 的问题都转到 networkErrors 吗?这不利于正确处理验证错误?
  • Fronters 说他们没有这样的麻烦。
【解决方案2】:

您可以尝试使用社区维护的软件包之一进行输入验证

https://github.com/ChilliCream/hotchocolate/blob/develop/COMMUNITY.md#validation

或通过提供的代码示例编写自己的代码

【讨论】:

  • 感谢您的回答。看起来它可以解决我的问题。我将在实现验证库后写入 cmets 结果
  • 嗨!我使用了其中一个库(一个名为 Graph.ArgumentValidation)。它帮助我验证了在查询正文中写入的输入,但不在 GraphQL 变量字段中。它在调用者堆栈上抛出异常 - 在解析部分(如果我理解正确的话)。问题是我无法填写错误的字段。它只说解析问题(例如:“Int can't parse given literal value of type Float”之类的东西)。我试图查看所有库的源代码,发现他们所有的中间件只有在解析失败后才捕获请求。
  • 我和经验更丰富的同事讨论了这个问题,他们说似乎没有我们需要的解决方案暂时不存在。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-12-15
相关资源
最近更新 更多