mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 03:22:36 +00:00 
			
		
		
		
	Sends email with information on the new user (time of creation and time of last sign-in) and a link to manage the new user from the admin panel closes: https://codeberg.org/forgejo/forgejo/issues/480 Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/1371 Co-authored-by: Aravinth Manivannan <realaravinth@batsense.net> Co-committed-by: Aravinth Manivannan <realaravinth@batsense.net> (cherry picked from commitc721aa828b) (cherry picked from commit6487efcb9d) Conflicts: modules/notification/base/notifier.go modules/notification/base/null.go modules/notification/notification.go https://codeberg.org/forgejo/forgejo/pulls/1422 (cherry picked from commit7ea66ee1c5) Conflicts: services/notify/notifier.go services/notify/notify.go services/notify/null.go https://codeberg.org/forgejo/forgejo/pulls/1469 (cherry picked from commit7d2d997011) (cherry picked from commit435a54f140) (cherry picked from commit8ec7b3e448) [GITEA] notifies admins on new user registration (squash) performance bottleneck Refs: https://codeberg.org/forgejo/forgejo/issues/1479 (cherry picked from commit97ac9147ff) (cherry picked from commit19f295c16b) (cherry picked from commit3367dcb2cf) [GITEA] notifies admins on new user registration (squash) cosmetic changes Co-authored-by: delvh <dev.lh@web.de> (cherry picked from commit9f1670e040) (cherry picked from commitde5bb2a224) (cherry picked from commit8f8e52f31a) (cherry picked from commite0d5130312) (cherry picked from commitf1288d6d9b) (cherry picked from commitf664f41658) (cherry picked from commite44e6c7e47) (cherry picked from commitc0d958cc4c) (cherry picked from commita88baa5e48) [GITEA] notifies admins on new user registration (squash) ctx.Locale (cherry picked from commit2f6329f693)
		
			
				
	
	
		
			556 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			556 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package user_test
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"math/rand"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"code.gitea.io/gitea/models/auth"
 | |
| 	"code.gitea.io/gitea/models/db"
 | |
| 	"code.gitea.io/gitea/models/unittest"
 | |
| 	user_model "code.gitea.io/gitea/models/user"
 | |
| 	"code.gitea.io/gitea/modules/auth/password/hash"
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/structs"
 | |
| 	"code.gitea.io/gitea/modules/timeutil"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| func TestOAuth2Application_LoadUser(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 	app := unittest.AssertExistsAndLoadBean(t, &auth.OAuth2Application{ID: 1})
 | |
| 	user, err := user_model.GetUserByID(db.DefaultContext, app.UID)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotNil(t, user)
 | |
| }
 | |
| 
 | |
| func TestGetUserEmailsByNames(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	// ignore none active user email
 | |
| 	assert.ElementsMatch(t, []string{"user8@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "user9"}))
 | |
| 	assert.ElementsMatch(t, []string{"user8@example.com", "user5@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "user5"}))
 | |
| 
 | |
| 	assert.ElementsMatch(t, []string{"user8@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "org7"}))
 | |
| }
 | |
| 
 | |
| func TestCanCreateOrganization(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	admin := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
 | |
| 	assert.True(t, admin.CanCreateOrganization())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	assert.True(t, user.CanCreateOrganization())
 | |
| 	// Disable user create organization permission.
 | |
| 	user.AllowCreateOrganization = false
 | |
| 	assert.False(t, user.CanCreateOrganization())
 | |
| 
 | |
| 	setting.Admin.DisableRegularOrgCreation = true
 | |
| 	user.AllowCreateOrganization = true
 | |
| 	assert.True(t, admin.CanCreateOrganization())
 | |
| 	assert.False(t, user.CanCreateOrganization())
 | |
| }
 | |
| 
 | |
| func TestSearchUsers(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 	testSuccess := func(opts *user_model.SearchUserOptions, expectedUserOrOrgIDs []int64) {
 | |
| 		users, _, err := user_model.SearchUsers(db.DefaultContext, opts)
 | |
| 		assert.NoError(t, err)
 | |
| 		cassText := fmt.Sprintf("ids: %v, opts: %v", expectedUserOrOrgIDs, opts)
 | |
| 		if assert.Len(t, users, len(expectedUserOrOrgIDs), "case: %s", cassText) {
 | |
| 			for i, expectedID := range expectedUserOrOrgIDs {
 | |
| 				assert.EqualValues(t, expectedID, users[i].ID, "case: %s", cassText)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// test orgs
 | |
| 	testOrgSuccess := func(opts *user_model.SearchUserOptions, expectedOrgIDs []int64) {
 | |
| 		opts.Type = user_model.UserTypeOrganization
 | |
| 		testSuccess(opts, expectedOrgIDs)
 | |
| 	}
 | |
| 
 | |
| 	testOrgSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 1, PageSize: 2}},
 | |
| 		[]int64{3, 6})
 | |
| 
 | |
| 	testOrgSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 2, PageSize: 2}},
 | |
| 		[]int64{7, 17})
 | |
| 
 | |
| 	testOrgSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 3, PageSize: 2}},
 | |
| 		[]int64{19, 25})
 | |
| 
 | |
| 	testOrgSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 4, PageSize: 2}},
 | |
| 		[]int64{26})
 | |
| 
 | |
| 	testOrgSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 5, PageSize: 2}},
 | |
| 		[]int64{})
 | |
| 
 | |
| 	// test users
 | |
| 	testUserSuccess := func(opts *user_model.SearchUserOptions, expectedUserIDs []int64) {
 | |
| 		opts.Type = user_model.UserTypeIndividual
 | |
| 		testSuccess(opts, expectedUserIDs)
 | |
| 	}
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 1}},
 | |
| 		[]int64{1, 2, 4, 5, 8, 9, 10, 11, 12, 13, 14, 15, 16, 18, 20, 21, 24, 27, 28, 29, 30, 32, 34})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 1}, IsActive: util.OptionalBoolFalse},
 | |
| 		[]int64{9})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 1}, IsActive: util.OptionalBoolTrue},
 | |
| 		[]int64{1, 2, 4, 5, 8, 10, 11, 12, 13, 14, 15, 16, 18, 20, 21, 24, 27, 28, 29, 30, 32, 34})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{Keyword: "user1", OrderBy: "id ASC", ListOptions: db.ListOptions{Page: 1}, IsActive: util.OptionalBoolTrue},
 | |
| 		[]int64{1, 10, 11, 12, 13, 14, 15, 16, 18})
 | |
| 
 | |
| 	// order by name asc default
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{Keyword: "user1", ListOptions: db.ListOptions{Page: 1}, IsActive: util.OptionalBoolTrue},
 | |
| 		[]int64{1, 10, 11, 12, 13, 14, 15, 16, 18})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 1}, IsAdmin: util.OptionalBoolTrue},
 | |
| 		[]int64{1})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 1}, IsRestricted: util.OptionalBoolTrue},
 | |
| 		[]int64{29})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 1}, IsProhibitLogin: util.OptionalBoolTrue},
 | |
| 		[]int64{30})
 | |
| 
 | |
| 	testUserSuccess(&user_model.SearchUserOptions{ListOptions: db.ListOptions{Page: 1}, IsTwoFactorEnabled: util.OptionalBoolTrue},
 | |
| 		[]int64{24})
 | |
| }
 | |
| 
 | |
| func TestEmailNotificationPreferences(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	for _, test := range []struct {
 | |
| 		expected string
 | |
| 		userID   int64
 | |
| 	}{
 | |
| 		{user_model.EmailNotificationsEnabled, 1},
 | |
| 		{user_model.EmailNotificationsEnabled, 2},
 | |
| 		{user_model.EmailNotificationsOnMention, 3},
 | |
| 		{user_model.EmailNotificationsOnMention, 4},
 | |
| 		{user_model.EmailNotificationsEnabled, 5},
 | |
| 		{user_model.EmailNotificationsEnabled, 6},
 | |
| 		{user_model.EmailNotificationsDisabled, 7},
 | |
| 		{user_model.EmailNotificationsEnabled, 8},
 | |
| 		{user_model.EmailNotificationsOnMention, 9},
 | |
| 	} {
 | |
| 		user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: test.userID})
 | |
| 		assert.Equal(t, test.expected, user.EmailNotifications())
 | |
| 
 | |
| 		// Try all possible settings
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(db.DefaultContext, user, user_model.EmailNotificationsEnabled))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsEnabled, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(db.DefaultContext, user, user_model.EmailNotificationsOnMention))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsOnMention, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(db.DefaultContext, user, user_model.EmailNotificationsDisabled))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsDisabled, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(db.DefaultContext, user, user_model.EmailNotificationsAndYourOwn))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsAndYourOwn, user.EmailNotifications())
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestHashPasswordDeterministic(t *testing.T) {
 | |
| 	b := make([]byte, 16)
 | |
| 	u := &user_model.User{}
 | |
| 	algos := hash.RecommendedHashAlgorithms
 | |
| 	for j := 0; j < len(algos); j++ {
 | |
| 		u.PasswdHashAlgo = algos[j]
 | |
| 		for i := 0; i < 50; i++ {
 | |
| 			// generate a random password
 | |
| 			rand.Read(b)
 | |
| 			pass := string(b)
 | |
| 
 | |
| 			// save the current password in the user - hash it and store the result
 | |
| 			u.SetPassword(pass)
 | |
| 			r1 := u.Passwd
 | |
| 
 | |
| 			// run again
 | |
| 			u.SetPassword(pass)
 | |
| 			r2 := u.Passwd
 | |
| 
 | |
| 			assert.NotEqual(t, r1, r2)
 | |
| 			assert.True(t, u.ValidatePassword(pass))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func BenchmarkHashPassword(b *testing.B) {
 | |
| 	// BenchmarkHashPassword ensures that it takes a reasonable amount of time
 | |
| 	// to hash a password - in order to protect from brute-force attacks.
 | |
| 	pass := "password1337"
 | |
| 	u := &user_model.User{Passwd: pass}
 | |
| 	b.ResetTimer()
 | |
| 	for i := 0; i < b.N; i++ {
 | |
| 		u.SetPassword(pass)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestNewGitSig(t *testing.T) {
 | |
| 	users := make([]*user_model.User, 0, 20)
 | |
| 	err := db.GetEngine(db.DefaultContext).Find(&users)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	for _, user := range users {
 | |
| 		sig := user.NewGitSig()
 | |
| 		assert.NotContains(t, sig.Name, "<")
 | |
| 		assert.NotContains(t, sig.Name, ">")
 | |
| 		assert.NotContains(t, sig.Name, "\n")
 | |
| 		assert.NotEqual(t, len(strings.TrimSpace(sig.Name)), 0)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDisplayName(t *testing.T) {
 | |
| 	users := make([]*user_model.User, 0, 20)
 | |
| 	err := db.GetEngine(db.DefaultContext).Find(&users)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	for _, user := range users {
 | |
| 		displayName := user.DisplayName()
 | |
| 		assert.Equal(t, strings.TrimSpace(displayName), displayName)
 | |
| 		if len(strings.TrimSpace(user.FullName)) == 0 {
 | |
| 			assert.Equal(t, user.Name, displayName)
 | |
| 		}
 | |
| 		assert.NotEqual(t, len(strings.TrimSpace(displayName)), 0)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestCreateUserInvalidEmail(t *testing.T) {
 | |
| 	user := &user_model.User{
 | |
| 		Name:               "GiteaBot",
 | |
| 		Email:              "GiteaBot@gitea.io\r\n",
 | |
| 		Passwd:             ";p['////..-++']",
 | |
| 		IsAdmin:            false,
 | |
| 		Theme:              setting.UI.DefaultTheme,
 | |
| 		MustChangePassword: false,
 | |
| 	}
 | |
| 
 | |
| 	err := user_model.CreateUser(db.DefaultContext, user)
 | |
| 	assert.Error(t, err)
 | |
| 	assert.True(t, user_model.IsErrEmailCharIsNotSupported(err))
 | |
| }
 | |
| 
 | |
| func TestCreateUserEmailAlreadyUsed(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 
 | |
| 	// add new user with user2's email
 | |
| 	user.Name = "testuser"
 | |
| 	user.LowerName = strings.ToLower(user.Name)
 | |
| 	user.ID = 0
 | |
| 	err := user_model.CreateUser(db.DefaultContext, user)
 | |
| 	assert.Error(t, err)
 | |
| 	assert.True(t, user_model.IsErrEmailAlreadyUsed(err))
 | |
| }
 | |
| 
 | |
| func TestCreateUserCustomTimestamps(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 
 | |
| 	// Add new user with a custom creation timestamp.
 | |
| 	var creationTimestamp timeutil.TimeStamp = 12345
 | |
| 	user.Name = "testuser"
 | |
| 	user.LowerName = strings.ToLower(user.Name)
 | |
| 	user.ID = 0
 | |
| 	user.Email = "unique@example.com"
 | |
| 	user.CreatedUnix = creationTimestamp
 | |
| 	err := user_model.CreateUser(db.DefaultContext, user)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	fetched, err := user_model.GetUserByID(context.Background(), user.ID)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Equal(t, creationTimestamp, fetched.CreatedUnix)
 | |
| 	assert.Equal(t, creationTimestamp, fetched.UpdatedUnix)
 | |
| }
 | |
| 
 | |
| func TestCreateUserWithoutCustomTimestamps(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 
 | |
| 	// There is no way to use a mocked time for the XORM auto-time functionality,
 | |
| 	// so use the real clock to approximate the expected timestamp.
 | |
| 	timestampStart := time.Now().Unix()
 | |
| 
 | |
| 	// Add new user without a custom creation timestamp.
 | |
| 	user.Name = "Testuser"
 | |
| 	user.LowerName = strings.ToLower(user.Name)
 | |
| 	user.ID = 0
 | |
| 	user.Email = "unique@example.com"
 | |
| 	user.CreatedUnix = 0
 | |
| 	user.UpdatedUnix = 0
 | |
| 	err := user_model.CreateUser(db.DefaultContext, user)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	timestampEnd := time.Now().Unix()
 | |
| 
 | |
| 	fetched, err := user_model.GetUserByID(context.Background(), user.ID)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	assert.LessOrEqual(t, timestampStart, fetched.CreatedUnix)
 | |
| 	assert.LessOrEqual(t, fetched.CreatedUnix, timestampEnd)
 | |
| 
 | |
| 	assert.LessOrEqual(t, timestampStart, fetched.UpdatedUnix)
 | |
| 	assert.LessOrEqual(t, fetched.UpdatedUnix, timestampEnd)
 | |
| }
 | |
| 
 | |
| func TestGetUserIDsByNames(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	// ignore non existing
 | |
| 	IDs, err := user_model.GetUserIDsByNames(db.DefaultContext, []string{"user1", "user2", "none_existing_user"}, true)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Equal(t, []int64{1, 2}, IDs)
 | |
| 
 | |
| 	// ignore non existing
 | |
| 	IDs, err = user_model.GetUserIDsByNames(db.DefaultContext, []string{"user1", "do_not_exist"}, false)
 | |
| 	assert.Error(t, err)
 | |
| 	assert.Equal(t, []int64(nil), IDs)
 | |
| }
 | |
| 
 | |
| func TestGetMaileableUsersByIDs(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	results, err := user_model.GetMaileableUsersByIDs(db.DefaultContext, []int64{1, 4}, false)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Len(t, results, 1)
 | |
| 	if len(results) > 1 {
 | |
| 		assert.Equal(t, results[0].ID, 1)
 | |
| 	}
 | |
| 
 | |
| 	results, err = user_model.GetMaileableUsersByIDs(db.DefaultContext, []int64{1, 4}, true)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Len(t, results, 2)
 | |
| 	if len(results) > 2 {
 | |
| 		assert.Equal(t, results[0].ID, 1)
 | |
| 		assert.Equal(t, results[1].ID, 4)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestUpdateUser(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 
 | |
| 	user.KeepActivityPrivate = true
 | |
| 	assert.NoError(t, user_model.UpdateUser(db.DefaultContext, user, false))
 | |
| 	user = unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	assert.True(t, user.KeepActivityPrivate)
 | |
| 
 | |
| 	setting.Service.AllowedUserVisibilityModesSlice = []bool{true, false, false}
 | |
| 	user.KeepActivityPrivate = false
 | |
| 	user.Visibility = structs.VisibleTypePrivate
 | |
| 	assert.Error(t, user_model.UpdateUser(db.DefaultContext, user, false))
 | |
| 	user = unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	assert.True(t, user.KeepActivityPrivate)
 | |
| 
 | |
| 	newEmail := "new_" + user.Email
 | |
| 	user.Email = newEmail
 | |
| 	assert.NoError(t, user_model.UpdateUser(db.DefaultContext, user, true))
 | |
| 	user = unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	assert.Equal(t, newEmail, user.Email)
 | |
| 
 | |
| 	user.Email = "no mail@mail.org"
 | |
| 	assert.Error(t, user_model.UpdateUser(db.DefaultContext, user, true))
 | |
| }
 | |
| 
 | |
| func TestUpdateUserEmailAlreadyUsed(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 	user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	org3 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 3})
 | |
| 
 | |
| 	user2.Email = org3.Email
 | |
| 	err := user_model.UpdateUser(db.DefaultContext, user2, true)
 | |
| 	assert.True(t, user_model.IsErrEmailAlreadyUsed(err))
 | |
| }
 | |
| 
 | |
| func TestNewUserRedirect(t *testing.T) {
 | |
| 	// redirect to a completely new name
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
 | |
| 	assert.NoError(t, user_model.NewUserRedirect(db.DefaultContext, user.ID, user.Name, "newusername"))
 | |
| 
 | |
| 	unittest.AssertExistsAndLoadBean(t, &user_model.Redirect{
 | |
| 		LowerName:      user.LowerName,
 | |
| 		RedirectUserID: user.ID,
 | |
| 	})
 | |
| 	unittest.AssertExistsAndLoadBean(t, &user_model.Redirect{
 | |
| 		LowerName:      "olduser1",
 | |
| 		RedirectUserID: user.ID,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestNewUserRedirect2(t *testing.T) {
 | |
| 	// redirect to previously used name
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
 | |
| 	assert.NoError(t, user_model.NewUserRedirect(db.DefaultContext, user.ID, user.Name, "olduser1"))
 | |
| 
 | |
| 	unittest.AssertExistsAndLoadBean(t, &user_model.Redirect{
 | |
| 		LowerName:      user.LowerName,
 | |
| 		RedirectUserID: user.ID,
 | |
| 	})
 | |
| 	unittest.AssertNotExistsBean(t, &user_model.Redirect{
 | |
| 		LowerName:      "olduser1",
 | |
| 		RedirectUserID: user.ID,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestNewUserRedirect3(t *testing.T) {
 | |
| 	// redirect for a previously-unredirected user
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
 | |
| 	assert.NoError(t, user_model.NewUserRedirect(db.DefaultContext, user.ID, user.Name, "newusername"))
 | |
| 
 | |
| 	unittest.AssertExistsAndLoadBean(t, &user_model.Redirect{
 | |
| 		LowerName:      user.LowerName,
 | |
| 		RedirectUserID: user.ID,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestGetUserByOpenID(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	_, err := user_model.GetUserByOpenID(db.DefaultContext, "https://unknown")
 | |
| 	if assert.Error(t, err) {
 | |
| 		assert.True(t, user_model.IsErrUserNotExist(err))
 | |
| 	}
 | |
| 
 | |
| 	user, err := user_model.GetUserByOpenID(db.DefaultContext, "https://user1.domain1.tld")
 | |
| 	if assert.NoError(t, err) {
 | |
| 		assert.Equal(t, int64(1), user.ID)
 | |
| 	}
 | |
| 
 | |
| 	user, err = user_model.GetUserByOpenID(db.DefaultContext, "https://domain1.tld/user2/")
 | |
| 	if assert.NoError(t, err) {
 | |
| 		assert.Equal(t, int64(2), user.ID)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestFollowUser(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	testSuccess := func(followerID, followedID int64) {
 | |
| 		assert.NoError(t, user_model.FollowUser(db.DefaultContext, followerID, followedID))
 | |
| 		unittest.AssertExistsAndLoadBean(t, &user_model.Follow{UserID: followerID, FollowID: followedID})
 | |
| 	}
 | |
| 	testSuccess(4, 2)
 | |
| 	testSuccess(5, 2)
 | |
| 
 | |
| 	assert.NoError(t, user_model.FollowUser(db.DefaultContext, 2, 2))
 | |
| 
 | |
| 	unittest.CheckConsistencyFor(t, &user_model.User{})
 | |
| }
 | |
| 
 | |
| func TestUnfollowUser(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	testSuccess := func(followerID, followedID int64) {
 | |
| 		assert.NoError(t, user_model.UnfollowUser(db.DefaultContext, followerID, followedID))
 | |
| 		unittest.AssertNotExistsBean(t, &user_model.Follow{UserID: followerID, FollowID: followedID})
 | |
| 	}
 | |
| 	testSuccess(4, 2)
 | |
| 	testSuccess(5, 2)
 | |
| 	testSuccess(2, 2)
 | |
| 
 | |
| 	unittest.CheckConsistencyFor(t, &user_model.User{})
 | |
| }
 | |
| 
 | |
| func TestIsUserVisibleToViewer(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	user1 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})   // admin, public
 | |
| 	user4 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 4})   // normal, public
 | |
| 	user20 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 20}) // public, same team as user31
 | |
| 	user29 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 29}) // public, is restricted
 | |
| 	user31 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 31}) // private, same team as user20
 | |
| 	user33 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 33}) // limited, follows 31
 | |
| 
 | |
| 	test := func(u, viewer *user_model.User, expected bool) {
 | |
| 		name := func(u *user_model.User) string {
 | |
| 			if u == nil {
 | |
| 				return "<nil>"
 | |
| 			}
 | |
| 			return u.Name
 | |
| 		}
 | |
| 		assert.Equal(t, expected, user_model.IsUserVisibleToViewer(db.DefaultContext, u, viewer), "user %v should be visible to viewer %v: %v", name(u), name(viewer), expected)
 | |
| 	}
 | |
| 
 | |
| 	// admin viewer
 | |
| 	test(user1, user1, true)
 | |
| 	test(user20, user1, true)
 | |
| 	test(user31, user1, true)
 | |
| 	test(user33, user1, true)
 | |
| 
 | |
| 	// non admin viewer
 | |
| 	test(user4, user4, true)
 | |
| 	test(user20, user4, true)
 | |
| 	test(user31, user4, false)
 | |
| 	test(user33, user4, true)
 | |
| 	test(user4, nil, true)
 | |
| 
 | |
| 	// public user
 | |
| 	test(user4, user20, true)
 | |
| 	test(user4, user31, true)
 | |
| 	test(user4, user33, true)
 | |
| 
 | |
| 	// limited user
 | |
| 	test(user33, user33, true)
 | |
| 	test(user33, user4, true)
 | |
| 	test(user33, user29, false)
 | |
| 	test(user33, nil, false)
 | |
| 
 | |
| 	// private user
 | |
| 	test(user31, user31, true)
 | |
| 	test(user31, user4, false)
 | |
| 	test(user31, user20, true)
 | |
| 	test(user31, user29, false)
 | |
| 	test(user31, user33, true)
 | |
| 	test(user31, nil, false)
 | |
| }
 | |
| 
 | |
| func Test_ValidateUser(t *testing.T) {
 | |
| 	oldSetting := setting.Service.AllowedUserVisibilityModesSlice
 | |
| 	defer func() {
 | |
| 		setting.Service.AllowedUserVisibilityModesSlice = oldSetting
 | |
| 	}()
 | |
| 	setting.Service.AllowedUserVisibilityModesSlice = []bool{true, false, true}
 | |
| 	kases := map[*user_model.User]bool{
 | |
| 		{ID: 1, Visibility: structs.VisibleTypePublic}:                            true,
 | |
| 		{ID: 2, Visibility: structs.VisibleTypeLimited}:                           false,
 | |
| 		{ID: 2, Visibility: structs.VisibleTypeLimited, Email: "invalid"}:         false,
 | |
| 		{ID: 2, Visibility: structs.VisibleTypePrivate, Email: "valid@valid.com"}: true,
 | |
| 	}
 | |
| 	for kase, expected := range kases {
 | |
| 		err := user_model.ValidateUser(kase)
 | |
| 		assert.EqualValues(t, expected, err == nil, fmt.Sprintf("case: %+v", kase))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestGetAllAdmins(t *testing.T) {
 | |
| 	assert.NoError(t, unittest.PrepareTestDatabase())
 | |
| 
 | |
| 	admins, err := user_model.GetAllAdmins(db.DefaultContext)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	assert.Len(t, admins, 1)
 | |
| 	assert.Equal(t, int64(1), admins[0].ID)
 | |
| }
 |