使用golang.org/x/oauth2进行golang Facebook身份验证

 微笑5885 发布于 2022-12-07 11:46

我正在尝试使用galang.org/x/oauth2包编写一个简单的程序.但我似乎无法交换访问令牌的代码.以下错误有点误导,因为它说已经使用了授权代码,但每次调用登录对话框时我都会看到一个新代码.我是Galang的新手,我可能会犯一个基本的错误,任何指针都会非常有用:)

    clientOptions, err = oauth2.New(
        oauth2.Client("xxxxxx", "22222222222222"),
        oauth2.RedirectURL("http://localhost:3000/auth/cb/fb2"),
        oauth2.Scope("public_profile", "email", "user_friends"),
        oauth2.Endpoint(
            "https://www.facebook.com/dialog/oauth",
            "https://graph.facebook.com/oauth/access_token",
        ),
    )

func handleFBSetupOauth(w http.ResponseWriter, r *http.Request) {
    url := clientOptions.AuthCodeURL("state", "online", "auto")
    fmt.Printf("Visit the URL for the auth dialog: %v", url)

    http.Redirect(w, r, url, http.StatusFound)
}

func handleFBOauthCB(w http.ResponseWriter, r *http.Request) (int, string) {
    var err error

    code := r.FormValue("code")
    if code == "" {
        return 500, "No code!"
    }

    fmt.Printf("code - %s", code)

    t, err := clientOptions.NewTransportFromCode(code)
    if err != nil {
        log.Fatal(err)
    }

    client := http.Client{Transport: t}

    url := "https://graph.facebook.com/oauth/access_token?client_id=xxxxxxx&redirect_uri=http://localhost:3000/auth/cb/fb2&client_secret=22222222&code=" + code + ""

    resp, err := client.Get(url)

我从上次获取请求中收到以下错误 -

{"error":{"message":"此授权码已被使用.","type":"OAuthException","code":100}}

我正在关注这两个指南 - Facebook登录流程 - https://developers.facebook.com/docs/facebook-login/manually-build-a-login-flow/v2.2

最新的goauth doc - https://godoc.org/golang.org/x/oauth2

1 个回答
  • 这是一个简单的例子来帮助您入门:

    package main
    
    import (
      "fmt"
      "io/ioutil"
      "log"
      "net/http"
      "net/url"
      "strings"
    
      "golang.org/x/oauth2"
      "golang.org/x/oauth2/facebook"
    )
    
    var (
      oauthConf = &oauth2.Config{
        ClientID:     "YOUR_CLIENT_ID",
        ClientSecret: "YOUR_CLIENT_SECRET",
        RedirectURL:  "YOUR_REDIRECT_URL_CALLBACK",
        Scopes:       []string{"public_profile"},
        Endpoint:     facebook.Endpoint,
      }
      oauthStateString = "thisshouldberandom"
    )
    
    const htmlIndex = `<html><body>
    Logged in with <a href="/login">facebook</a>
    </body></html>
    `
    
    func handleMain(w http.ResponseWriter, r *http.Request) {
      w.Header().Set("Content-Type", "text/html; charset=utf-8")
      w.WriteHeader(http.StatusOK)
      w.Write([]byte(htmlIndex))
    }
    
    func handleFacebookLogin(w http.ResponseWriter, r *http.Request) {
      Url, err := url.Parse(oauthConf.Endpoint.AuthURL)
      if err != nil {
        log.Fatal("Parse: ", err)
      }
      parameters := url.Values{}
      parameters.Add("client_id", oauthConf.ClientID)
      parameters.Add("scope", strings.Join(oauthConf.Scopes, " "))
      parameters.Add("redirect_uri", oauthConf.RedirectURL)
      parameters.Add("response_type", "code")
      parameters.Add("state", oauthStateString)
      Url.RawQuery = parameters.Encode()
      url := Url.String()
      http.Redirect(w, r, url, http.StatusTemporaryRedirect)
    }
    
    func handleFacebookCallback(w http.ResponseWriter, r *http.Request) {
      state := r.FormValue("state")
      if state != oauthStateString {
        fmt.Printf("invalid oauth state, expected '%s', got '%s'\n", oauthStateString, state)
        http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
        return
      }
    
      code := r.FormValue("code")
    
      token, err := oauthConf.Exchange(oauth2.NoContext, code)
      if err != nil {
        fmt.Printf("oauthConf.Exchange() failed with '%s'\n", err)
        http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
        return
      }
    
      resp, err := http.Get("https://graph.facebook.com/me?access_token=" +
        url.QueryEscape(token.AccessToken))
      if err != nil {
        fmt.Printf("Get: %s\n", err)
        http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
        return
      }
      defer resp.Body.Close()
    
      response, err := ioutil.ReadAll(resp.Body)
      if err != nil {
        fmt.Printf("ReadAll: %s\n", err)
        http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
        return
      }
    
      log.Printf("parseResponseBody: %s\n", string(response))
    
      http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
    }
    
    func main() {
      http.HandleFunc("/", handleMain)
      http.HandleFunc("/login", handleFacebookLogin)
      http.HandleFunc("/oauth2callback", handleFacebookCallback)
      fmt.Print("Started running on http://localhost:9090\n")
      log.Fatal(http.ListenAndServe(":9090", nil))
    }
    

    2022-12-11 02:56 回答
撰写答案
今天,你开发时遇到什么问题呢?
立即提问
热门标签
PHP1.CN | 中国最专业的PHP中文社区 | PNG素材下载 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有