Gabe Kangas b80ccc4966
WIP
2024-03-25 09:04:05 -07:00

132 lines
4.4 KiB
Go

package fediverse
import (
"encoding/json"
"fmt"
"net/http"
"github.com/owncast/owncast/models"
"github.com/owncast/owncast/services/apfederation"
fediverseauth "github.com/owncast/owncast/services/auth/fediverse"
"github.com/owncast/owncast/services/chat"
"github.com/owncast/owncast/storage/chatrepository"
"github.com/owncast/owncast/storage/configrepository"
"github.com/owncast/owncast/storage/userrepository"
"github.com/owncast/owncast/webserver/responses"
log "github.com/sirupsen/logrus"
)
type FediAuthHandlers struct {
configRepository *configrepository.SqlConfigRepository
chatService *chat.Chat
chatRepository *chatrepository.ChatRepository
userRepository *userrepository.SqlUserRepository
}
// New creates a new instances of web server handlers.
func New() *FediAuthHandlers {
return &FediAuthHandlers{
configRepository: configrepository.Get(),
chatService: chat.Get(),
chatRepository: chatrepository.Get(),
userRepository: userrepository.Get(),
}
}
// RegisterFediverseOTPRequest registers a new OTP request for the given access token.
func (h *FediAuthHandlers) RegisterFediverseOTPRequest(u models.User, w http.ResponseWriter, r *http.Request) {
type request struct {
FediverseAccount string `json:"account"`
}
var req request
decoder := json.NewDecoder(r.Body)
if err := decoder.Decode(&req); err != nil {
responses.WriteSimpleResponse(w, false, "Could not decode request: "+err.Error())
return
}
fediAuth := fediverseauth.GetFediAuth()
accessToken := r.URL.Query().Get("accessToken")
reg, success, err := fediAuth.RegisterFediverseOTP(accessToken, u.ID, u.DisplayName, req.FediverseAccount)
if err != nil {
responses.WriteSimpleResponse(w, false, "Could not register auth request: "+err.Error())
return
}
if !success {
responses.WriteSimpleResponse(w, false, "Could not register auth request. One may already be pending. Try again later.")
return
}
activitypub := apfederation.Get()
msg := fmt.Sprintf("<p>This is an automated message from %s. If you did not request this message please ignore or block. Your requested one-time code is:</p><p>%s</p>", h.configRepository.GetServerName(), reg.Code)
if err := activitypub.SendDirectFederatedMessage(msg, reg.Account); err != nil {
responses.WriteSimpleResponse(w, false, "Could not send code to fediverse: "+err.Error())
return
}
responses.WriteSimpleResponse(w, true, "")
}
// VerifyFediverseOTPRequest verifies the given OTP code for the given access token.
func (h *FediAuthHandlers) VerifyFediverseOTPRequest(w http.ResponseWriter, r *http.Request) {
type request struct {
Code string `json:"code"`
}
var req request
decoder := json.NewDecoder(r.Body)
if err := decoder.Decode(&req); err != nil {
responses.WriteSimpleResponse(w, false, "Could not decode request: "+err.Error())
return
}
accessToken := r.URL.Query().Get("accessToken")
fediAuth := fediverseauth.GetFediAuth()
valid, authRegistration := fediAuth.ValidateFediverseOTP(accessToken, req.Code)
if !valid {
w.WriteHeader(http.StatusForbidden)
return
}
// Check if a user with this auth already exists, if so, log them in.
if u := h.userRepository.GetUserByAuth(authRegistration.Account, models.Fediverse); u != nil {
// Handle existing auth.
log.Debugln("user with provided fedvierse identity already exists, logging them in")
// Update the current user's access token to point to the existing user id.
userID := u.ID
if err := h.userRepository.SetAccessTokenToOwner(accessToken, userID); err != nil {
responses.WriteSimpleResponse(w, false, err.Error())
return
}
if authRegistration.UserDisplayName != u.DisplayName {
loginMessage := fmt.Sprintf("**%s** is now authenticated as **%s**", authRegistration.UserDisplayName, u.DisplayName)
if err := h.chatService.SendSystemAction(loginMessage, true); err != nil {
log.Errorln(err)
}
}
responses.WriteSimpleResponse(w, true, "")
return
}
// Otherwise, save this as new auth.
log.Debug("fediverse account does not already exist, saving it as a new one for the current user")
if err := h.userRepository.AddAuth(authRegistration.UserID, authRegistration.Account, models.Fediverse); err != nil {
responses.WriteSimpleResponse(w, false, err.Error())
return
}
// Update the current user's authenticated flag so we can show it in
// the chat UI.
if err := h.userRepository.SetUserAsAuthenticated(authRegistration.UserID); err != nil {
log.Errorln(err)
}
responses.WriteSimpleResponse(w, true, "")
}