aboutsummaryrefslogtreecommitdiff
path: root/internal/db
diff options
context:
space:
mode:
authorJoe Chen <jc@unknwon.io>2022-10-30 23:04:24 +0800
committerGitHub <noreply@github.com>2022-10-30 23:04:24 +0800
commit1905b19ee7fcc5622ad3ad4d85cc8fc2e81a0b56 (patch)
tree11c6c7bd33837733167f5f3adc9c09dacfbdd19f /internal/db
parent131be6e074039e590488892d7a99ebdbe6eb4668 (diff)
refactor(db): migrate methods off `user.go` (#7219)
Diffstat (limited to 'internal/db')
-rw-r--r--internal/db/access_tokens_test.go2
-rw-r--r--internal/db/actions_test.go2
-rw-r--r--internal/db/db.go1
-rw-r--r--internal/db/follows.go4
-rw-r--r--internal/db/follows_test.go2
-rw-r--r--internal/db/lfs_test.go2
-rw-r--r--internal/db/login_sources_test.go4
-rw-r--r--internal/db/org.go20
-rw-r--r--internal/db/org_users.go38
-rw-r--r--internal/db/org_users_test.go51
-rw-r--r--internal/db/perms_test.go2
-rw-r--r--internal/db/repos_test.go2
-rw-r--r--internal/db/two_factors_test.go2
-rw-r--r--internal/db/user.go49
-rw-r--r--internal/db/users.go36
-rw-r--r--internal/db/users_test.go2
-rw-r--r--internal/db/watches_test.go2
17 files changed, 146 insertions, 75 deletions
diff --git a/internal/db/access_tokens_test.go b/internal/db/access_tokens_test.go
index a14f9a72..2c915ab8 100644
--- a/internal/db/access_tokens_test.go
+++ b/internal/db/access_tokens_test.go
@@ -105,7 +105,7 @@ func TestAccessTokens(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *accessTokens)
+ test func(t *testing.T, db *accessTokens)
}{
{"Create", accessTokensCreate},
{"DeleteByID", accessTokensDeleteByID},
diff --git a/internal/db/actions_test.go b/internal/db/actions_test.go
index 9c8dae37..cc64850e 100644
--- a/internal/db/actions_test.go
+++ b/internal/db/actions_test.go
@@ -106,7 +106,7 @@ func TestActions(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *actions)
+ test func(t *testing.T, db *actions)
}{
{"CommitRepo", actionsCommitRepo},
{"ListByOrganization", actionsListByOrganization},
diff --git a/internal/db/db.go b/internal/db/db.go
index b765dfd8..d291fc49 100644
--- a/internal/db/db.go
+++ b/internal/db/db.go
@@ -124,6 +124,7 @@ func Init(w logger.Writer) (*gorm.DB, error) {
Follows = NewFollowsStore(db)
LoginSources = &loginSources{DB: db, files: sourceFiles}
LFS = &lfs{DB: db}
+ OrgUsers = NewOrgUsersStore(db)
Perms = &perms{DB: db}
Repos = NewReposStore(db)
TwoFactors = &twoFactors{DB: db}
diff --git a/internal/db/follows.go b/internal/db/follows.go
index 4f3d55f0..83f37e2e 100644
--- a/internal/db/follows.go
+++ b/internal/db/follows.go
@@ -11,7 +11,7 @@ import (
"gorm.io/gorm"
)
-// FollowsStore is the persistent interface for follows.
+// FollowsStore is the persistent interface for user follows.
//
// NOTE: All methods are sorted in alphabetical order.
type FollowsStore interface {
@@ -31,7 +31,7 @@ type follows struct {
*gorm.DB
}
-// NewFollowsStore returns a persistent interface for follows with given
+// NewFollowsStore returns a persistent interface for user follows with given
// database connection.
func NewFollowsStore(db *gorm.DB) FollowsStore {
return &follows{DB: db}
diff --git a/internal/db/follows_test.go b/internal/db/follows_test.go
index cd37cc97..4caa447e 100644
--- a/internal/db/follows_test.go
+++ b/internal/db/follows_test.go
@@ -27,7 +27,7 @@ func TestFollows(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *follows)
+ test func(t *testing.T, db *follows)
}{
{"Follow", followsFollow},
{"IsFollowing", followsIsFollowing},
diff --git a/internal/db/lfs_test.go b/internal/db/lfs_test.go
index a9a2a2bb..690962e7 100644
--- a/internal/db/lfs_test.go
+++ b/internal/db/lfs_test.go
@@ -30,7 +30,7 @@ func TestLFS(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *lfs)
+ test func(t *testing.T, db *lfs)
}{
{"CreateObject", lfsCreateObject},
{"GetObjectByOID", lfsGetObjectByOID},
diff --git a/internal/db/login_sources_test.go b/internal/db/login_sources_test.go
index f455761f..af0da97b 100644
--- a/internal/db/login_sources_test.go
+++ b/internal/db/login_sources_test.go
@@ -157,7 +157,7 @@ func TestLoginSource_AfterFind(t *testing.T) {
}
}
-func Test_loginSources(t *testing.T) {
+func TestLoginSources(t *testing.T) {
if testing.Short() {
t.Skip()
}
@@ -170,7 +170,7 @@ func Test_loginSources(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *loginSources)
+ test func(t *testing.T, db *loginSources)
}{
{"Create", loginSourcesCreate},
{"Count", loginSourcesCount},
diff --git a/internal/db/org.go b/internal/db/org.go
index 93350746..d3148745 100644
--- a/internal/db/org.go
+++ b/internal/db/org.go
@@ -235,14 +235,14 @@ func DeleteOrganization(org *User) (err error) {
// \_______ /__| \___ /|______//____ >\___ >__|
// \/ /_____/ \/ \/
-// OrgUser represents an organization-user relation.
+// OrgUser represents relations of organizations and their members.
type OrgUser struct {
- ID int64
- Uid int64 `xorm:"INDEX UNIQUE(s)"`
- OrgID int64 `xorm:"INDEX UNIQUE(s)"`
- IsPublic bool
- IsOwner bool
- NumTeams int
+ ID int64 `gorm:"primaryKey"`
+ Uid int64 `xorm:"INDEX UNIQUE(s)" gorm:"uniqueIndex:org_user_user_org_unique;index;not null"`
+ OrgID int64 `xorm:"INDEX UNIQUE(s)" gorm:"uniqueIndex:org_user_user_org_unique;index;not null"`
+ IsPublic bool `gorm:"not null;default:FALSE"`
+ IsOwner bool `gorm:"not null;default:FALSE"`
+ NumTeams int `gorm:"not null;default:0"`
}
// IsOrganizationOwner returns true if given user is in the owner team.
@@ -278,12 +278,6 @@ func GetOrgsByUserID(userID int64, showAll bool) ([]*User, error) {
return getOrgsByUserID(x.NewSession(), userID, showAll)
}
-// GetOrgsByUserIDDesc returns a list of organizations that the given user ID
-// has joined, ordered descending by the given condition.
-func GetOrgsByUserIDDesc(userID int64, desc string, showAll bool) ([]*User, error) {
- return getOrgsByUserID(x.NewSession().Desc(desc), userID, showAll)
-}
-
func getOwnedOrgsByUserID(sess *xorm.Session, userID int64) ([]*User, error) {
orgs := make([]*User, 0, 10)
return orgs, sess.Where("`org_user`.uid=?", userID).And("`org_user`.is_owner=?", true).
diff --git a/internal/db/org_users.go b/internal/db/org_users.go
new file mode 100644
index 00000000..5c4add26
--- /dev/null
+++ b/internal/db/org_users.go
@@ -0,0 +1,38 @@
+// Copyright 2022 The Gogs Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package db
+
+import (
+ "context"
+
+ "gorm.io/gorm"
+)
+
+// OrgUsersStore is the persistent interface for organization-user relations.
+//
+// NOTE: All methods are sorted in alphabetical order.
+type OrgUsersStore interface {
+ // CountByUser returns the number of organizations the user is a member of.
+ CountByUser(ctx context.Context, userID int64) (int64, error)
+}
+
+var OrgUsers OrgUsersStore
+
+var _ OrgUsersStore = (*orgUsers)(nil)
+
+type orgUsers struct {
+ *gorm.DB
+}
+
+// NewOrgUsersStore returns a persistent interface for organization-user
+// relations with given database connection.
+func NewOrgUsersStore(db *gorm.DB) OrgUsersStore {
+ return &orgUsers{DB: db}
+}
+
+func (db *orgUsers) CountByUser(ctx context.Context, userID int64) (int64, error) {
+ var count int64
+ return count, db.WithContext(ctx).Model(&OrgUser{}).Where("uid = ?", userID).Count(&count).Error
+}
diff --git a/internal/db/org_users_test.go b/internal/db/org_users_test.go
new file mode 100644
index 00000000..f5c486db
--- /dev/null
+++ b/internal/db/org_users_test.go
@@ -0,0 +1,51 @@
+// Copyright 2022 The Gogs Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package db
+
+import (
+ "testing"
+
+ "github.com/stretchr/testify/require"
+
+ "gogs.io/gogs/internal/dbtest"
+)
+
+func TestOrgUsers(t *testing.T) {
+ if testing.Short() {
+ t.Skip()
+ }
+ t.Parallel()
+
+ tables := []interface{}{new(OrgUser)}
+ db := &orgUsers{
+ DB: dbtest.NewDB(t, "orgUsers", tables...),
+ }
+
+ for _, tc := range []struct {
+ name string
+ test func(t *testing.T, db *orgUsers)
+ }{
+ {"CountByUser", orgUsersCountByUser},
+ } {
+ t.Run(tc.name, func(t *testing.T) {
+ t.Cleanup(func() {
+ err := clearTables(t, db.DB, tables...)
+ require.NoError(t, err)
+ })
+ tc.test(t, db)
+ })
+ if t.Failed() {
+ break
+ }
+ }
+}
+
+func orgUsersCountByUser(t *testing.T, db *orgUsers) {
+ // TODO: Use OrgUsers.Join to replace SQL hack when the method is available.
+ err := db.Exec(`INSERT INTO org_user (uid, org_id) VALUES (?, ?)`, 1, 1).Error
+ require.NoError(t, err)
+ err = db.Exec(`INSERT INTO org_user (uid, org_id) VALUES (?, ?)`, 2, 1).Error
+ require.NoError(t, err)
+}
diff --git a/internal/db/perms_test.go b/internal/db/perms_test.go
index 6c4e1478..1e09f1d2 100644
--- a/internal/db/perms_test.go
+++ b/internal/db/perms_test.go
@@ -27,7 +27,7 @@ func TestPerms(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *perms)
+ test func(t *testing.T, db *perms)
}{
{"AccessMode", permsAccessMode},
{"Authorize", permsAuthorize},
diff --git a/internal/db/repos_test.go b/internal/db/repos_test.go
index 2af42e8b..ef832389 100644
--- a/internal/db/repos_test.go
+++ b/internal/db/repos_test.go
@@ -92,7 +92,7 @@ func TestRepos(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *repos)
+ test func(t *testing.T, db *repos)
}{
{"Create", reposCreate},
{"GetByName", reposGetByName},
diff --git a/internal/db/two_factors_test.go b/internal/db/two_factors_test.go
index ddff3fce..e2d58cd5 100644
--- a/internal/db/two_factors_test.go
+++ b/internal/db/two_factors_test.go
@@ -74,7 +74,7 @@ func TestTwoFactors(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *twoFactors)
+ test func(t *testing.T, db *twoFactors)
}{
{"Create", twoFactorsCreate},
{"GetByUserID", twoFactorsGetByUserID},
diff --git a/internal/db/user.go b/internal/db/user.go
index 3fff95d4..ea947224 100644
--- a/internal/db/user.go
+++ b/internal/db/user.go
@@ -53,37 +53,13 @@ func (u *User) AfterSet(colName string, _ xorm.Cell) {
}
}
+// Deprecated: Use OrgsUsers.CountByUser instead.
+//
+// TODO(unknwon): Delete me once no more call sites.
func (u *User) getOrganizationCount(e Engine) (int64, error) {
return e.Where("uid=?", u.ID).Count(new(OrgUser))
}
-// GetOrganizationCount returns count of membership of organization of user.
-func (u *User) GetOrganizationCount() (int64, error) {
- return u.getOrganizationCount(x)
-}
-
-// GetRepositories returns repositories that user owns, including private repositories.
-func (u *User) GetRepositories(page, pageSize int) (err error) {
- u.Repos, err = GetUserRepositories(&UserRepoOptions{
- UserID: u.ID,
- Private: true,
- Page: page,
- PageSize: pageSize,
- })
- return err
-}
-
-// GetRepositories returns mirror repositories that user owns, including private repositories.
-func (u *User) GetMirrorRepositories() ([]*Repository, error) {
- return GetUserMirrorRepositories(u.ID)
-}
-
-// GetOwnedOrganizations returns all organizations that user owns.
-func (u *User) GetOwnedOrganizations() (err error) {
- u.OwnedOrgs, err = GetOwnedOrgsByUserID(u.ID)
- return err
-}
-
// GetOrganizations returns all organizations that user belongs to.
func (u *User) GetOrganizations(showPrivate bool) error {
orgIDs, err := GetOrgIDsByUserID(u.ID, showPrivate)
@@ -101,25 +77,6 @@ func (u *User) GetOrganizations(showPrivate bool) error {
return nil
}
-// DisplayName returns full name if it's not empty,
-// returns username otherwise.
-func (u *User) DisplayName() string {
- if len(u.FullName) > 0 {
- return u.FullName
- }
- return u.Name
-}
-
-func (u *User) ShortName(length int) string {
- return strutil.Ellipsis(u.Name, length)
-}
-
-// IsMailable checks if a user is eligible
-// to receive emails.
-func (u *User) IsMailable() bool {
- return u.IsActive
-}
-
// IsUserExist checks if given user name exist,
// the user name should be noncased unique.
// If uid is presented, then check will rule out that one,
diff --git a/internal/db/users.go b/internal/db/users.go
index bdd6501b..af0db727 100644
--- a/internal/db/users.go
+++ b/internal/db/users.go
@@ -22,6 +22,7 @@ import (
"gogs.io/gogs/internal/cryptoutil"
"gogs.io/gogs/internal/errutil"
"gogs.io/gogs/internal/osutil"
+ "gogs.io/gogs/internal/strutil"
"gogs.io/gogs/internal/tool"
"gogs.io/gogs/internal/userutil"
)
@@ -451,9 +452,7 @@ type User struct {
LoginSource int64 `xorm:"NOT NULL DEFAULT 0" gorm:"not null;default:0"`
LoginName string
Type UserType
- OwnedOrgs []*User `xorm:"-" gorm:"-" json:"-"`
- Orgs []*User `xorm:"-" gorm:"-" json:"-"`
- Repos []*Repository `xorm:"-" gorm:"-" json:"-"`
+ Orgs []*User `xorm:"-" gorm:"-" json:"-"`
Location string
Website string
Rands string `xorm:"VARCHAR(10)" gorm:"type:VARCHAR(10)"`
@@ -521,6 +520,11 @@ func (u *User) IsOrganization() bool {
return u.Type == UserTypeOrganization
}
+// IsMailable returns true if the user is eligible to receive emails.
+func (u *User) IsMailable() bool {
+ return u.IsActive
+}
+
// APIFormat returns the API format of a user.
func (u *User) APIFormat() *api.User {
return &api.User{
@@ -562,6 +566,15 @@ func (u *User) CanImportLocal() bool {
return conf.Repository.EnableLocalPathMigration && (u.IsAdmin || u.AllowImportLocal)
}
+// DisplayName returns the full name of the user if it's not empty, returns the
+// username otherwise.
+func (u *User) DisplayName() string {
+ if len(u.FullName) > 0 {
+ return u.FullName
+ }
+ return u.Name
+}
+
// HomeURLPath returns the URL path to the user or organization home page.
//
// TODO(unknwon): This is also used in templates, which should be fixed by
@@ -649,3 +662,20 @@ func (u *User) IsUserOrgOwner(orgId int64) bool {
func (u *User) IsPublicMember(orgId int64) bool {
return IsPublicMembership(orgId, u.ID)
}
+
+// GetOrganizationCount returns the count of organization membership that the
+// user has.
+//
+// TODO(unknwon): This is also used in templates, which should be fixed by
+// having a dedicated type `template.User`.
+func (u *User) GetOrganizationCount() (int64, error) {
+ return OrgUsers.CountByUser(context.TODO(), u.ID)
+}
+
+// ShortName truncates and returns the username at most in given length.
+//
+// TODO(unknwon): This is also used in templates, which should be fixed by
+// having a dedicated type `template.User`.
+func (u *User) ShortName(length int) string {
+ return strutil.Ellipsis(u.Name, length)
+}
diff --git a/internal/db/users_test.go b/internal/db/users_test.go
index 5645ff7b..6d71ad5a 100644
--- a/internal/db/users_test.go
+++ b/internal/db/users_test.go
@@ -84,7 +84,7 @@ func TestUsers(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *users)
+ test func(t *testing.T, db *users)
}{
{"Authenticate", usersAuthenticate},
{"Create", usersCreate},
diff --git a/internal/db/watches_test.go b/internal/db/watches_test.go
index 7ec5b93c..46267ccd 100644
--- a/internal/db/watches_test.go
+++ b/internal/db/watches_test.go
@@ -25,7 +25,7 @@ func TestWatches(t *testing.T) {
for _, tc := range []struct {
name string
- test func(*testing.T, *watches)
+ test func(t *testing.T, db *watches)
}{
{"ListByRepo", watchesListByRepo},
} {