如何从Golang OAuth2中正确读取错误

时间:2019-03-05 19:31:23

标签: go oauth oauth-2.0 godoc

token, err := googleOauthConfig.Exchange(context.Background(), code)
if err != nil {
 fmt.Fprintf(w, "Err: %+v", err)
}

fprintf的输出为:

Err: oauth2: cannot fetch token: 401 Unauthorized
Response: {"error":"code_already_used","error_description":"code_already_used"}

我要检查“ error”是否为“ code_already_used”。对于我的一生,我无法弄清楚怎么做。

如何检查/返回/读取错误的“错误”或“错误说明”?

我看过oauth2代码,它比我高一点。

// retrieveToken takes a *Config and uses that to retrieve an *internal.Token.
// This token is then mapped from *internal.Token into an *oauth2.Token which is returned along
// with an error..
func retrieveToken(ctx context.Context, c *Config, v url.Values) (*Token, error) {
    tk, err := internal.RetrieveToken(ctx, c.ClientID, c.ClientSecret, c.Endpoint.TokenURL, v)
    if err != nil {
        if rErr, ok := err.(*internal.RetrieveError); ok {
            return nil, (*RetrieveError)(rErr)
        }
        return nil, err
    }
    return tokenFromInternal(tk), nil
}

如何猜测我正在尝试查看(* RetrieveError)部分。对吧?

谢谢!

2 个答案:

答案 0 :(得分:1)

表达式:

(*RetrieveError)(rErr)

rErr的类型从*internal.RetrieveError转换为*RetrieveError。而且由于RetrieveError是在oauth2包中声明的,因此您可以键入assert *oauth2.RetrieveError收到的错误以获取详细信息。详细信息以字节的形式包含在该类型的“正文”字段中。

由于字节的片段不是要检查的最佳格式,并且在您的情况下,看起来字节包含json对象,因此可以通过预定义可将这些详细信息解组的类型来简化生活。

也就是说:

type ErrorDetails struct {
    Error            string `json:"error"`
    ErrorDescription string `json:"error_description"`
}

token, err := googleOauthConfig.Exchange(context.Background(), code)
if err != nil {
    fmt.Fprintf(w, "Err: %+v", err)

    if rErr, ok := err.(*oauth2.RetrieveError); ok {
        details := new(ErrorDetails)
        if err := json.Unmarshal(rErr.Body, details); err != nil {
            panic(err)
        }

        fmt.Println(details.Error, details.ErrorDescription)
    }        
}

答案 1 :(得分:0)

可以这样做。


arr := strings.Split(err.Error(), "\n")
        str := strings.Replace(arr[1], "Response: ", "", 1)
        var details ErrorDetails
        var json = jsoniter.ConfigCompatibleWithStandardLibrary

        err := json.Unmarshal([]byte(str), &details)
        if err == nil {
            beego.Debug(details.Error)
            beego.Debug(details.ErrorDescription)
        }