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("
This is an automated message from %s. If you did not request this message please ignore or block. Your requested one-time code is:
%s
", 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, "") }