mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-10 09:05:30 +00:00
f7638f5414
user, topic, project, label, milestone, repository, pull_request, release, asset, comment, reaction, review providers Signed-off-by: Earl Warren <contact@earl-warren.org> Preserve file size when creating attachments Introduced inc6f5029708
repoList.LoadAttributes has a ctx argument now Rename `repo.GetOwner` to `repo.LoadOwner`bd66fa586a
upgrade to the latest gof3 (cherry picked from commitc770713656
) [F3] ID remapping logic is in place, remove workaround (cherry picked from commitd0fee30167
) [F3] it is experimental, do not enable by default (cherry picked from commitde325b21d0
) (cherry picked from commit547e7b3c40
) (cherry picked from commit820df3a56b
) (cherry picked from commiteaba87689b
) (cherry picked from commit1b86896b3b
) (cherry picked from commit0046aac1c6
) (cherry picked from commitf14220df8f
) (cherry picked from commit559b731001
) (cherry picked from commit801f7d600d
) (cherry picked from commit6aa76e9bcf
) (cherry picked from commita8757dcb07
) [F3] promote F3 users to matching OAuth2 users on first sign-in (cherry picked from commitbd7fef7496
) (cherry picked from commit07412698e8
) (cherry picked from commitd143e5b2a3
) [F3] upgrade to gof3 50a6e740ac04 Add new methods GetIDString() & SetIDString() & ToFormatInterface() Change the prototype of the fixture function (cherry picked from commitd7b263ff8b
) (cherry picked from commitb3eaf2249d
) (cherry picked from commitd492ddd9bb
) [F3] add GetLocalMatchingRemote with a default implementation (cherry picked from commit0a22015039
) (cherry picked from commitf1310c38fb
) (cherry picked from commitdeb68552f2
) [F3] GetLocalMatchingRemote for user (cherry picked from commite73cb837f5
) (cherry picked from commita24bc0b85e
) (cherry picked from commit846a522ecc
) [F3] GetAdminUser now has a ctx argument (cherry picked from commit37357a92af
) (cherry picked from commit660bc1673c
) (cherry picked from commit72d692a767
) [F3] introduce UserTypeF3 To avoid conflicts should UserTypeRemoteUser be used differently by Gitea (cherry picked from commit6de2701bb3
) [F3] user.Put: idempotency (cherry picked from commit821e38573c
)
115 lines
3.8 KiB
Go
115 lines
3.8 KiB
Go
// SPDX-FileCopyrightText: Copyright the Forgejo contributors
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package f3
|
|
|
|
import (
|
|
"context"
|
|
|
|
auth_model "code.gitea.io/gitea/models/auth"
|
|
"code.gitea.io/gitea/models/db"
|
|
user_model "code.gitea.io/gitea/models/user"
|
|
"code.gitea.io/gitea/modules/log"
|
|
f3_source "code.gitea.io/gitea/services/auth/source/f3"
|
|
"code.gitea.io/gitea/services/auth/source/oauth2"
|
|
)
|
|
|
|
func getUserByLoginName(ctx context.Context, name string) (*user_model.User, error) {
|
|
if len(name) == 0 {
|
|
return nil, user_model.ErrUserNotExist{Name: name}
|
|
}
|
|
u := &user_model.User{LoginName: name, LoginType: auth_model.F3, Type: user_model.UserTypeF3}
|
|
has, err := db.GetEngine(ctx).Get(u)
|
|
if err != nil {
|
|
return nil, err
|
|
} else if !has {
|
|
return nil, user_model.ErrUserNotExist{Name: name}
|
|
}
|
|
return u, nil
|
|
}
|
|
|
|
// The user created by F3 has:
|
|
//
|
|
// Type UserTypeF3
|
|
// LogingType F3
|
|
// LoginName set to the unique identifier of the originating forge
|
|
// LoginSource set to the F3 source that can be matched against a OAuth2 source
|
|
//
|
|
// If the source from which an authentification happens is OAuth2, an existing
|
|
// F3 user will be promoted to an OAuth2 user provided:
|
|
//
|
|
// user.LoginName is the same as goth.UserID (argument loginName)
|
|
// user.LoginSource has a MatchingSource equals to the name of the OAuth2 provider
|
|
//
|
|
// Once promoted, the user will be logged in without further interaction from the
|
|
// user and will own all repositories, issues, etc. associated with it.
|
|
func MaybePromoteF3User(ctx context.Context, source *auth_model.Source, loginName, email string) error {
|
|
user, err := getF3UserToPromote(ctx, source, loginName, email)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if user != nil {
|
|
promote := &user_model.User{
|
|
ID: user.ID,
|
|
Type: user_model.UserTypeIndividual,
|
|
Email: email,
|
|
LoginSource: source.ID,
|
|
LoginType: source.Type,
|
|
}
|
|
log.Debug("promote user %v: LoginName %v => %v, LoginSource %v => %v, LoginType %v => %v, Email %v => %v", user.ID, user.LoginName, promote.LoginName, user.LoginSource, promote.LoginSource, user.LoginType, promote.LoginType, user.Email, promote.Email)
|
|
return user_model.UpdateUser(ctx, promote, true, "type", "email", "login_source", "login_type")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func getF3UserToPromote(ctx context.Context, source *auth_model.Source, loginName, email string) (*user_model.User, error) {
|
|
if !source.IsOAuth2() {
|
|
log.Debug("getF3UserToPromote: source %v is not OAuth2", source)
|
|
return nil, nil
|
|
}
|
|
oauth2Source, ok := source.Cfg.(*oauth2.Source)
|
|
if !ok {
|
|
log.Error("getF3UserToPromote: source claims to be OAuth2 but really is %v", oauth2Source)
|
|
return nil, nil
|
|
}
|
|
|
|
u, err := getUserByLoginName(ctx, loginName)
|
|
if err != nil {
|
|
if user_model.IsErrUserNotExist(err) {
|
|
log.Debug("getF3UserToPromote: no user with LoginType F3 and LoginName '%s'", loginName)
|
|
return nil, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if !u.IsF3() {
|
|
log.Debug("getF3UserToPromote: user %v is not a managed by F3", u)
|
|
return nil, nil
|
|
}
|
|
|
|
if u.Email != "" {
|
|
log.Debug("getF3UserToPromote: the user email is already set to '%s'", u.Email)
|
|
return nil, nil
|
|
}
|
|
|
|
userSource, err := auth_model.GetSourceByID(u.LoginSource)
|
|
if err != nil {
|
|
if auth_model.IsErrSourceNotExist(err) {
|
|
log.Error("getF3UserToPromote: source id = %v for user %v not found %v", u.LoginSource, u.ID, err)
|
|
return nil, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
f3Source, ok := userSource.Cfg.(*f3_source.Source)
|
|
if !ok {
|
|
log.Error("getF3UserToPromote: expected an F3 source but got %T %v", userSource, userSource)
|
|
return nil, nil
|
|
}
|
|
|
|
if oauth2Source.Provider != f3Source.MatchingSource {
|
|
log.Debug("getF3UserToPromote: skip OAuth2 source %s because it is different from %s which is the expected match for the F3 source %s", oauth2Source.Provider, f3Source.MatchingSource, f3Source.URL)
|
|
return nil, nil
|
|
}
|
|
|
|
return u, nil
|
|
}
|