mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 17:08:25 +00:00 
			
		
		
		
	This PR refactors and improves the password hashing code within gitea and makes it possible for server administrators to set the password hashing parameters In addition it takes the opportunity to adjust the settings for `pbkdf2` in order to make the hashing a little stronger. The majority of this work was inspired by PR #14751 and I would like to thank @boppy for their work on this. Thanks to @gusted for the suggestion to adjust the `pbkdf2` hashing parameters. Close #14751 --------- Signed-off-by: Andrew Thornton <art27@cantab.net> Co-authored-by: delvh <dev.lh@web.de> Co-authored-by: John Olheiser <john.olheiser@gmail.com> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
		
			
				
	
	
		
			527 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			527 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package user_test
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"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.Equal(t, []string{"user8@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "user9"}))
 | |
| 	assert.Equal(t, []string{"user8@example.com", "user5@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "user5"}))
 | |
| 
 | |
| 	assert.Equal(t, []string{"user8@example.com"}, user_model.GetUserEmailsByNames(db.DefaultContext, []string{"user8", "user7"}))
 | |
| }
 | |
| 
 | |
| 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(opts)
 | |
| 		assert.NoError(t, err)
 | |
| 		if assert.Len(t, users, len(expectedUserOrOrgIDs), opts) {
 | |
| 			for i, expectedID := range expectedUserOrOrgIDs {
 | |
| 				assert.EqualValues(t, expectedID, users[i].ID)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// 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})
 | |
| 
 | |
| 	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})
 | |
| 
 | |
| 	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, 30})
 | |
| 
 | |
| 	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(user, user_model.EmailNotificationsEnabled))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsEnabled, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(user, user_model.EmailNotificationsOnMention))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsOnMention, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(user, user_model.EmailNotificationsDisabled))
 | |
| 		assert.Equal(t, user_model.EmailNotificationsDisabled, user.EmailNotifications())
 | |
| 
 | |
| 		assert.NoError(t, user_model.SetEmailNotifications(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(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(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(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(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})
 | |
| 	user3 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 3})
 | |
| 
 | |
| 	user2.Email = user3.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("https://unknown")
 | |
| 	if assert.Error(t, err) {
 | |
| 		assert.True(t, user_model.IsErrUserNotExist(err))
 | |
| 	}
 | |
| 
 | |
| 	user, err := user_model.GetUserByOpenID("https://user1.domain1.tld")
 | |
| 	if assert.NoError(t, err) {
 | |
| 		assert.Equal(t, int64(1), user.ID)
 | |
| 	}
 | |
| 
 | |
| 	user, err = user_model.GetUserByOpenID("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(followerID, followedID))
 | |
| 		unittest.AssertExistsAndLoadBean(t, &user_model.Follow{UserID: followerID, FollowID: followedID})
 | |
| 	}
 | |
| 	testSuccess(4, 2)
 | |
| 	testSuccess(5, 2)
 | |
| 
 | |
| 	assert.NoError(t, user_model.FollowUser(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(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)
 | |
| }
 |