statping/handlers/oauth.go

177 lines
3.9 KiB
Go
Raw Normal View History

2020-04-07 11:53:32 +00:00
package handlers
import (
"encoding/json"
2020-04-07 11:53:32 +00:00
"fmt"
"github.com/gorilla/mux"
"github.com/statping/statping/types/core"
"github.com/statping/statping/types/null"
"github.com/statping/statping/types/users"
"github.com/statping/statping/utils"
2020-04-07 11:53:32 +00:00
"golang.org/x/oauth2"
"golang.org/x/oauth2/github"
"golang.org/x/oauth2/google"
2020-04-07 14:56:49 +00:00
"golang.org/x/oauth2/slack"
2020-04-07 11:53:32 +00:00
"net/http"
"time"
2020-04-07 11:53:32 +00:00
)
type oAuth struct {
ID string
2020-04-07 11:53:32 +00:00
Email string
Username string
2020-04-07 11:53:32 +00:00
Token string
RefreshToken string
Valid bool
Type string
2020-04-07 11:53:32 +00:00
}
func oauthHandler(w http.ResponseWriter, r *http.Request) {
vars := mux.Vars(r)
provider := vars["provider"]
code := r.URL.Query().Get("code")
fmt.Println("code: ", code)
fmt.Println("client: ", core.App.OAuth.SlackClientID)
fmt.Println("secret: ", core.App.OAuth.SlackClientSecret)
2020-04-07 11:53:32 +00:00
var err error
var oauth *oAuth
switch provider {
case "google":
oauth, err = googleOAuth(r)
2020-04-07 11:53:32 +00:00
case "github":
oauth, err = githubOAuth(r)
2020-04-20 03:02:34 +00:00
case "slack":
oauth, err = slackOAuth(r)
2020-04-07 11:53:32 +00:00
}
if err != nil {
log.Error(err)
2020-04-20 03:02:34 +00:00
sendErrorJson(err, w, r)
2020-04-07 11:53:32 +00:00
return
}
oauthLogin(oauth, w, r)
}
func oauthLogin(oauth *oAuth, w http.ResponseWriter, r *http.Request) {
2020-04-20 03:02:34 +00:00
log.Infoln(oauth)
2020-04-07 11:53:32 +00:00
user := &users.User{
Id: 0,
Username: oauth.Username,
2020-04-07 11:53:32 +00:00
Email: oauth.Email,
Admin: null.NewNullBool(true),
}
log.Infoln(fmt.Sprintf("OAuth User %s logged in from IP %s", oauth.Email, r.RemoteAddr))
2020-04-07 11:53:32 +00:00
setJwtToken(user, w)
//returnJson(user, w, r)
http.Redirect(w, r, core.App.Domain, http.StatusPermanentRedirect)
2020-04-07 11:53:32 +00:00
}
func githubOAuth(r *http.Request) (*oAuth, error) {
2020-04-07 11:53:32 +00:00
c := *core.App
code := r.URL.Query().Get("code")
config := &oauth2.Config{
ClientID: c.OAuth.GithubClientID,
ClientSecret: c.OAuth.GithubClientSecret,
Endpoint: github.Endpoint,
}
gg, err := config.Exchange(r.Context(), code)
if err != nil {
return nil, err
2020-04-07 11:53:32 +00:00
}
return &oAuth{
2020-04-07 11:53:32 +00:00
Token: gg.AccessToken,
RefreshToken: gg.RefreshToken,
Valid: gg.Valid(),
}, nil
2020-04-07 11:53:32 +00:00
}
func googleOAuth(r *http.Request) (*oAuth, error) {
2020-04-20 03:02:34 +00:00
c := core.App
2020-04-07 11:53:32 +00:00
code := r.URL.Query().Get("code")
config := &oauth2.Config{
ClientID: c.OAuth.GoogleClientID,
ClientSecret: c.OAuth.GoogleClientSecret,
2020-04-07 11:53:32 +00:00
Endpoint: google.Endpoint,
}
gg, err := config.Exchange(r.Context(), code)
if err != nil {
return nil, err
2020-04-07 11:53:32 +00:00
}
return &oAuth{
2020-04-07 11:53:32 +00:00
Token: gg.AccessToken,
RefreshToken: gg.RefreshToken,
Valid: gg.Valid(),
}, nil
2020-04-07 11:53:32 +00:00
}
2020-04-07 14:56:49 +00:00
func slackOAuth(r *http.Request) (*oAuth, error) {
c := core.App
2020-04-07 14:56:49 +00:00
code := r.URL.Query().Get("code")
config := &oauth2.Config{
ClientID: c.OAuth.SlackClientID,
ClientSecret: c.OAuth.SlackClientSecret,
2020-04-07 14:56:49 +00:00
Endpoint: slack.Endpoint,
RedirectURL: c.Domain + basePath + "oauth/slack",
Scopes: []string{"identity.basic"},
2020-04-07 14:56:49 +00:00
}
gg, err := config.Exchange(r.Context(), code)
if err != nil {
return nil, err
2020-04-07 14:56:49 +00:00
}
oauther := &oAuth{
2020-04-07 14:56:49 +00:00
Token: gg.AccessToken,
RefreshToken: gg.RefreshToken,
Valid: gg.Valid(),
Type: gg.Type(),
2020-04-07 14:56:49 +00:00
}
return oauther.slackIdentity()
}
// slackIdentity will query the Slack API to fetch the users ID, username, and email address.
func (a *oAuth) slackIdentity() (*oAuth, error) {
url := fmt.Sprintf("https://slack.com/api/users.identity?token=%s", a.Token)
out, resp, err := utils.HttpRequest(url, "GET", "application/x-www-form-urlencoded", nil, nil, 10*time.Second, true)
if err != nil {
return a, err
}
defer resp.Body.Close()
var i *slackIdentity
if err := json.Unmarshal(out, &i); err != nil {
return a, err
}
a.Email = i.User.Email
a.ID = i.User.ID
a.Username = i.User.Name
return a, nil
}
type slackIdentity struct {
Ok bool `json:"ok"`
User struct {
Name string `json:"name"`
ID string `json:"id"`
Email string `json:"email"`
} `json:"user"`
Team struct {
ID string `json:"id"`
} `json:"team"`
2020-04-07 14:56:49 +00:00
}
func secureToken(w http.ResponseWriter, r *http.Request) {
2020-04-08 03:30:41 +00:00
2020-04-07 14:56:49 +00:00
}