Как правильно читать ошибки от golang oauth2 - PullRequest
0 голосов
/ 05 марта 2019
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".Что касается жизни, я не могу разобраться, как.

Как мне проверить / вернуть / прочитать "error" или "error_description" из err?

Я посмотрел на код 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 ]

1 голос
/ 05 марта 2019

Выражение:

(*RetrieveError)(rErr)

преобразует тип rErr из *internal.RetrieveError в *RetrieveError. И поскольку RetrieveError объявлено в пакете oauth2, вы можете напечатать assert полученную ошибку, чтобы *oauth2.RetrieveError получить подробности. Детали содержатся в поле Body этого типа в виде фрагмента байтов.

Поскольку фрагмент байтов - не лучший формат для проверки, и в вашем случае кажется, что байты содержат объект 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)
    }        
}
0 голосов
/ 03 июля 2019

Можно сделать так.


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)
        }

...