refactor(db): migrate avatar methods off `user.go` (#7206)

pull/7207/head
Joe Chen 2022-10-23 20:54:16 +08:00 committed by GitHub
parent c58c893621
commit d0a4a3401c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 431 additions and 77 deletions

View File

@ -14,11 +14,11 @@ import (
"github.com/issue9/identicon"
)
const AVATAR_SIZE = 290
const DefaultSize = 290
// RandomImage generates and returns a random avatar image unique to input data
// in custom size (height and width).
func RandomImageSize(size int, data []byte) (image.Image, error) {
// RandomImageWithSize generates and returns a random avatar image unique to
// input data in custom size (height and width).
func RandomImageWithSize(size int, data []byte) (image.Image, error) {
randExtent := len(palette.WebSafe) - 32
rand.Seed(time.Now().UnixNano())
colorIndex := rand.Intn(randExtent)
@ -37,7 +37,7 @@ func RandomImageSize(size int, data []byte) (image.Image, error) {
}
// RandomImage generates and returns a random avatar image unique to input data
// in default size (height and width).
// in DefaultSize (height and width).
func RandomImage(data []byte) (image.Image, error) {
return RandomImageSize(AVATAR_SIZE, data)
return RandomImageWithSize(DefaultSize, data)
}

View File

@ -12,10 +12,7 @@ import (
func Test_RandomImage(t *testing.T) {
_, err := RandomImage([]byte("gogs@local"))
if err != nil {
t.Fatal(err)
}
_, err = RandomImageSize(0, []byte("gogs@local"))
assert.NoError(t, err)
_, err = RandomImageWithSize(0, []byte("gogs@local"))
assert.Error(t, err)
}

View File

@ -336,7 +336,7 @@ func (repo *Repository) UploadAvatar(data []byte) error {
}
defer fw.Close()
m := resize.Resize(avatar.AVATAR_SIZE, avatar.AVATAR_SIZE, img, resize.NearestNeighbor)
m := resize.Resize(avatar.DefaultSize, avatar.DefaultSize, img, resize.NearestNeighbor)
if err = png.Encode(fw, m); err != nil {
return fmt.Errorf("encode image: %v", err)
}

View File

@ -5,27 +5,22 @@
package db
import (
"bytes"
"context"
"encoding/hex"
"fmt"
"image"
_ "image/jpeg"
"image/png"
"os"
"path/filepath"
"strings"
"time"
"unicode/utf8"
"github.com/nfnt/resize"
"github.com/unknwon/com"
log "unknwon.dev/clog/v2"
"xorm.io/xorm"
"github.com/gogs/git-module"
"gogs.io/gogs/internal/avatar"
"gogs.io/gogs/internal/conf"
"gogs.io/gogs/internal/db/errors"
"gogs.io/gogs/internal/errutil"
@ -58,41 +53,6 @@ func (u *User) AfterSet(colName string, _ xorm.Cell) {
}
}
// UploadAvatar saves custom avatar for user.
// FIXME: split uploads to different subdirs in case we have massive number of users.
func (u *User) UploadAvatar(data []byte) error {
img, _, err := image.Decode(bytes.NewReader(data))
if err != nil {
return fmt.Errorf("decode image: %v", err)
}
_ = os.MkdirAll(conf.Picture.AvatarUploadPath, os.ModePerm)
fw, err := os.Create(userutil.CustomAvatarPath(u.ID))
if err != nil {
return fmt.Errorf("create custom avatar directory: %v", err)
}
defer fw.Close()
m := resize.Resize(avatar.AVATAR_SIZE, avatar.AVATAR_SIZE, img, resize.NearestNeighbor)
if err = png.Encode(fw, m); err != nil {
return fmt.Errorf("encode image: %v", err)
}
return nil
}
// DeleteAvatar deletes the user's custom avatar.
func (u *User) DeleteAvatar() error {
avatarPath := userutil.CustomAvatarPath(u.ID)
log.Trace("DeleteAvatar [%d]: %s", u.ID, avatarPath)
if err := os.Remove(avatarPath); err != nil {
return err
}
u.UseCustomAvatar = false
return UpdateUser(u)
}
// IsAdminOfRepo returns true if user has admin or higher access of repository.
func (u *User) IsAdminOfRepo(repo *Repository) bool {
return Perms.Authorize(context.TODO(), u.ID, repo.ID, AccessModeAdmin,

View File

@ -7,6 +7,7 @@ package db
import (
"context"
"fmt"
"os"
"strings"
"time"
@ -45,6 +46,9 @@ type UsersStore interface {
// ErrUserAlreadyExist when a user with same name already exists, or
// ErrEmailAlreadyUsed if the email has been used by another user.
Create(ctx context.Context, username, email string, opts CreateUserOptions) (*User, error)
// DeleteCustomAvatar deletes the current user custom avatar and falls back to
// use look up avatar by email.
DeleteCustomAvatar(ctx context.Context, userID int64) error
// GetByEmail returns the user (not organization) with given email. It ignores
// records with unverified emails and returns ErrUserNotExist when not found.
GetByEmail(ctx context.Context, email string) (*User, error)
@ -64,6 +68,8 @@ type UsersStore interface {
// Results are paginated by given page and page size, and sorted by the time of
// follow in descending order.
ListFollowings(ctx context.Context, userID int64, page, pageSize int) ([]*User, error)
// UseCustomAvatar uses the given avatar as the user custom avatar.
UseCustomAvatar(ctx context.Context, userID int64, avatar []byte) error
}
var Users UsersStore
@ -267,6 +273,18 @@ func (db *users) Create(ctx context.Context, username, email string, opts Create
return user, db.WithContext(ctx).Create(user).Error
}
func (db *users) DeleteCustomAvatar(ctx context.Context, userID int64) error {
_ = os.Remove(userutil.CustomAvatarPath(userID))
return db.WithContext(ctx).
Model(&User{}).
Where("id = ?", userID).
Updates(map[string]interface{}{
"use_custom_avatar": false,
"updated_unix": db.NowFunc().Unix(),
}).
Error
}
var _ errutil.NotFound = (*ErrUserNotExist)(nil)
type ErrUserNotExist struct {
@ -397,6 +415,22 @@ func (db *users) ListFollowings(ctx context.Context, userID int64, page, pageSiz
return users, tx.Find(&users).Error
}
func (db *users) UseCustomAvatar(ctx context.Context, userID int64, avatar []byte) error {
err := userutil.SaveAvatar(userID, avatar)
if err != nil {
return errors.Wrap(err, "save avatar")
}
return db.WithContext(ctx).
Model(&User{}).
Where("id = ?", userID).
Updates(map[string]interface{}{
"use_custom_avatar": true,
"updated_unix": db.NowFunc().Unix(),
}).
Error
}
// UserType indicates the type of the user account.
type UserType int

View File

@ -7,6 +7,7 @@ package db
import (
"context"
"fmt"
"os"
"testing"
"time"
@ -16,6 +17,9 @@ import (
"gogs.io/gogs/internal/auth"
"gogs.io/gogs/internal/dbtest"
"gogs.io/gogs/internal/errutil"
"gogs.io/gogs/internal/osutil"
"gogs.io/gogs/internal/userutil"
"gogs.io/gogs/public"
)
func TestUsers(t *testing.T) {
@ -35,12 +39,14 @@ func TestUsers(t *testing.T) {
}{
{"Authenticate", usersAuthenticate},
{"Create", usersCreate},
{"DeleteCustomAvatar", usersDeleteCustomAvatar},
{"GetByEmail", usersGetByEmail},
{"GetByID", usersGetByID},
{"GetByUsername", usersGetByUsername},
{"HasForkedRepository", usersHasForkedRepository},
{"ListFollowers", usersListFollowers},
{"ListFollowings", usersListFollowings},
{"UseCustomAvatar", usersUseCustomAvatar},
} {
t.Run(tc.name, func(t *testing.T) {
t.Cleanup(func() {
@ -186,6 +192,42 @@ func usersCreate(t *testing.T, db *users) {
assert.Equal(t, db.NowFunc().Format(time.RFC3339), user.Updated.UTC().Format(time.RFC3339))
}
func usersDeleteCustomAvatar(t *testing.T, db *users) {
ctx := context.Background()
alice, err := db.Create(ctx, "alice", "alice@example.com", CreateUserOptions{})
require.NoError(t, err)
avatar, err := public.Files.ReadFile("img/avatar_default.png")
require.NoError(t, err)
avatarPath := userutil.CustomAvatarPath(alice.ID)
_ = os.Remove(avatarPath)
defer func() { _ = os.Remove(avatarPath) }()
err = db.UseCustomAvatar(ctx, alice.ID, avatar)
require.NoError(t, err)
// Make sure avatar is saved and the user flag is updated.
got := osutil.IsFile(avatarPath)
assert.True(t, got)
alice, err = db.GetByID(ctx, alice.ID)
require.NoError(t, err)
assert.True(t, alice.UseCustomAvatar)
// Delete avatar should remove the file and revert the user flag.
err = db.DeleteCustomAvatar(ctx, alice.ID)
require.NoError(t, err)
got = osutil.IsFile(avatarPath)
assert.False(t, got)
alice, err = db.GetByID(ctx, alice.ID)
require.NoError(t, err)
assert.False(t, alice.UseCustomAvatar)
}
func usersGetByEmail(t *testing.T, db *users) {
ctx := context.Background()
@ -366,3 +408,28 @@ func usersListFollowings(t *testing.T, db *users) {
require.Len(t, got, 1)
assert.Equal(t, alice.ID, got[0].ID)
}
func usersUseCustomAvatar(t *testing.T, db *users) {
ctx := context.Background()
alice, err := db.Create(ctx, "alice", "alice@example.com", CreateUserOptions{})
require.NoError(t, err)
avatar, err := public.Files.ReadFile("img/avatar_default.png")
require.NoError(t, err)
avatarPath := userutil.CustomAvatarPath(alice.ID)
_ = os.Remove(avatarPath)
defer func() { _ = os.Remove(avatarPath) }()
err = db.UseCustomAvatar(ctx, alice.ID, avatar)
require.NoError(t, err)
// Make sure avatar is saved and the user flag is updated.
got := osutil.IsFile(avatarPath)
assert.True(t, got)
alice, err = db.GetByID(ctx, alice.ID)
require.NoError(t, err)
assert.True(t, alice.UseCustomAvatar)
}

View File

@ -2299,6 +2299,9 @@ type MockUsersStore struct {
// CreateFunc is an instance of a mock function object controlling the
// behavior of the method Create.
CreateFunc *UsersStoreCreateFunc
// DeleteCustomAvatarFunc is an instance of a mock function object
// controlling the behavior of the method DeleteCustomAvatar.
DeleteCustomAvatarFunc *UsersStoreDeleteCustomAvatarFunc
// GetByEmailFunc is an instance of a mock function object controlling
// the behavior of the method GetByEmail.
GetByEmailFunc *UsersStoreGetByEmailFunc
@ -2317,6 +2320,9 @@ type MockUsersStore struct {
// ListFollowingsFunc is an instance of a mock function object
// controlling the behavior of the method ListFollowings.
ListFollowingsFunc *UsersStoreListFollowingsFunc
// UseCustomAvatarFunc is an instance of a mock function object
// controlling the behavior of the method UseCustomAvatar.
UseCustomAvatarFunc *UsersStoreUseCustomAvatarFunc
}
// NewMockUsersStore creates a new mock of the UsersStore interface. All
@ -2333,6 +2339,11 @@ func NewMockUsersStore() *MockUsersStore {
return
},
},
DeleteCustomAvatarFunc: &UsersStoreDeleteCustomAvatarFunc{
defaultHook: func(context.Context, int64) (r0 error) {
return
},
},
GetByEmailFunc: &UsersStoreGetByEmailFunc{
defaultHook: func(context.Context, string) (r0 *db.User, r1 error) {
return
@ -2363,6 +2374,11 @@ func NewMockUsersStore() *MockUsersStore {
return
},
},
UseCustomAvatarFunc: &UsersStoreUseCustomAvatarFunc{
defaultHook: func(context.Context, int64, []byte) (r0 error) {
return
},
},
}
}
@ -2380,6 +2396,11 @@ func NewStrictMockUsersStore() *MockUsersStore {
panic("unexpected invocation of MockUsersStore.Create")
},
},
DeleteCustomAvatarFunc: &UsersStoreDeleteCustomAvatarFunc{
defaultHook: func(context.Context, int64) error {
panic("unexpected invocation of MockUsersStore.DeleteCustomAvatar")
},
},
GetByEmailFunc: &UsersStoreGetByEmailFunc{
defaultHook: func(context.Context, string) (*db.User, error) {
panic("unexpected invocation of MockUsersStore.GetByEmail")
@ -2410,6 +2431,11 @@ func NewStrictMockUsersStore() *MockUsersStore {
panic("unexpected invocation of MockUsersStore.ListFollowings")
},
},
UseCustomAvatarFunc: &UsersStoreUseCustomAvatarFunc{
defaultHook: func(context.Context, int64, []byte) error {
panic("unexpected invocation of MockUsersStore.UseCustomAvatar")
},
},
}
}
@ -2423,6 +2449,9 @@ func NewMockUsersStoreFrom(i db.UsersStore) *MockUsersStore {
CreateFunc: &UsersStoreCreateFunc{
defaultHook: i.Create,
},
DeleteCustomAvatarFunc: &UsersStoreDeleteCustomAvatarFunc{
defaultHook: i.DeleteCustomAvatar,
},
GetByEmailFunc: &UsersStoreGetByEmailFunc{
defaultHook: i.GetByEmail,
},
@ -2441,6 +2470,9 @@ func NewMockUsersStoreFrom(i db.UsersStore) *MockUsersStore {
ListFollowingsFunc: &UsersStoreListFollowingsFunc{
defaultHook: i.ListFollowings,
},
UseCustomAvatarFunc: &UsersStoreUseCustomAvatarFunc{
defaultHook: i.UseCustomAvatar,
},
}
}
@ -2671,6 +2703,112 @@ func (c UsersStoreCreateFuncCall) Results() []interface{} {
return []interface{}{c.Result0, c.Result1}
}
// UsersStoreDeleteCustomAvatarFunc describes the behavior when the
// DeleteCustomAvatar method of the parent MockUsersStore instance is
// invoked.
type UsersStoreDeleteCustomAvatarFunc struct {
defaultHook func(context.Context, int64) error
hooks []func(context.Context, int64) error
history []UsersStoreDeleteCustomAvatarFuncCall
mutex sync.Mutex
}
// DeleteCustomAvatar delegates to the next hook function in the queue and
// stores the parameter and result values of this invocation.
func (m *MockUsersStore) DeleteCustomAvatar(v0 context.Context, v1 int64) error {
r0 := m.DeleteCustomAvatarFunc.nextHook()(v0, v1)
m.DeleteCustomAvatarFunc.appendCall(UsersStoreDeleteCustomAvatarFuncCall{v0, v1, r0})
return r0
}
// SetDefaultHook sets function that is called when the DeleteCustomAvatar
// method of the parent MockUsersStore instance is invoked and the hook
// queue is empty.
func (f *UsersStoreDeleteCustomAvatarFunc) SetDefaultHook(hook func(context.Context, int64) error) {
f.defaultHook = hook
}
// PushHook adds a function to the end of hook queue. Each invocation of the
// DeleteCustomAvatar method of the parent MockUsersStore instance invokes
// the hook at the front of the queue and discards it. After the queue is
// empty, the default hook function is invoked for any future action.
func (f *UsersStoreDeleteCustomAvatarFunc) PushHook(hook func(context.Context, int64) error) {
f.mutex.Lock()
f.hooks = append(f.hooks, hook)
f.mutex.Unlock()
}
// SetDefaultReturn calls SetDefaultHook with a function that returns the
// given values.
func (f *UsersStoreDeleteCustomAvatarFunc) SetDefaultReturn(r0 error) {
f.SetDefaultHook(func(context.Context, int64) error {
return r0
})
}
// PushReturn calls PushHook with a function that returns the given values.
func (f *UsersStoreDeleteCustomAvatarFunc) PushReturn(r0 error) {
f.PushHook(func(context.Context, int64) error {
return r0
})
}
func (f *UsersStoreDeleteCustomAvatarFunc) nextHook() func(context.Context, int64) error {
f.mutex.Lock()
defer f.mutex.Unlock()
if len(f.hooks) == 0 {
return f.defaultHook
}
hook := f.hooks[0]
f.hooks = f.hooks[1:]
return hook
}
func (f *UsersStoreDeleteCustomAvatarFunc) appendCall(r0 UsersStoreDeleteCustomAvatarFuncCall) {
f.mutex.Lock()
f.history = append(f.history, r0)
f.mutex.Unlock()
}
// History returns a sequence of UsersStoreDeleteCustomAvatarFuncCall
// objects describing the invocations of this function.
func (f *UsersStoreDeleteCustomAvatarFunc) History() []UsersStoreDeleteCustomAvatarFuncCall {
f.mutex.Lock()
history := make([]UsersStoreDeleteCustomAvatarFuncCall, len(f.history))
copy(history, f.history)
f.mutex.Unlock()
return history
}
// UsersStoreDeleteCustomAvatarFuncCall is an object that describes an
// invocation of method DeleteCustomAvatar on an instance of MockUsersStore.
type UsersStoreDeleteCustomAvatarFuncCall struct {
// Arg0 is the value of the 1st argument passed to this method
// invocation.
Arg0 context.Context
// Arg1 is the value of the 2nd argument passed to this method
// invocation.
Arg1 int64
// Result0 is the value of the 1st result returned from this method
// invocation.
Result0 error
}
// Args returns an interface slice containing the arguments of this
// invocation.
func (c UsersStoreDeleteCustomAvatarFuncCall) Args() []interface{} {
return []interface{}{c.Arg0, c.Arg1}
}
// Results returns an interface slice containing the results of this
// invocation.
func (c UsersStoreDeleteCustomAvatarFuncCall) Results() []interface{} {
return []interface{}{c.Result0}
}
// UsersStoreGetByEmailFunc describes the behavior when the GetByEmail
// method of the parent MockUsersStore instance is invoked.
type UsersStoreGetByEmailFunc struct {
@ -3332,3 +3470,111 @@ func (c UsersStoreListFollowingsFuncCall) Args() []interface{} {
func (c UsersStoreListFollowingsFuncCall) Results() []interface{} {
return []interface{}{c.Result0, c.Result1}
}
// UsersStoreUseCustomAvatarFunc describes the behavior when the
// UseCustomAvatar method of the parent MockUsersStore instance is invoked.
type UsersStoreUseCustomAvatarFunc struct {
defaultHook func(context.Context, int64, []byte) error
hooks []func(context.Context, int64, []byte) error
history []UsersStoreUseCustomAvatarFuncCall
mutex sync.Mutex
}
// UseCustomAvatar delegates to the next hook function in the queue and
// stores the parameter and result values of this invocation.
func (m *MockUsersStore) UseCustomAvatar(v0 context.Context, v1 int64, v2 []byte) error {
r0 := m.UseCustomAvatarFunc.nextHook()(v0, v1, v2)
m.UseCustomAvatarFunc.appendCall(UsersStoreUseCustomAvatarFuncCall{v0, v1, v2, r0})
return r0
}
// SetDefaultHook sets function that is called when the UseCustomAvatar
// method of the parent MockUsersStore instance is invoked and the hook
// queue is empty.
func (f *UsersStoreUseCustomAvatarFunc) SetDefaultHook(hook func(context.Context, int64, []byte) error) {
f.defaultHook = hook
}
// PushHook adds a function to the end of hook queue. Each invocation of the
// UseCustomAvatar method of the parent MockUsersStore instance invokes the
// hook at the front of the queue and discards it. After the queue is empty,
// the default hook function is invoked for any future action.
func (f *UsersStoreUseCustomAvatarFunc) PushHook(hook func(context.Context, int64, []byte) error) {
f.mutex.Lock()
f.hooks = append(f.hooks, hook)
f.mutex.Unlock()
}
// SetDefaultReturn calls SetDefaultHook with a function that returns the
// given values.
func (f *UsersStoreUseCustomAvatarFunc) SetDefaultReturn(r0 error) {
f.SetDefaultHook(func(context.Context, int64, []byte) error {
return r0
})
}
// PushReturn calls PushHook with a function that returns the given values.
func (f *UsersStoreUseCustomAvatarFunc) PushReturn(r0 error) {
f.PushHook(func(context.Context, int64, []byte) error {
return r0
})
}
func (f *UsersStoreUseCustomAvatarFunc) nextHook() func(context.Context, int64, []byte) error {
f.mutex.Lock()
defer f.mutex.Unlock()
if len(f.hooks) == 0 {
return f.defaultHook
}
hook := f.hooks[0]
f.hooks = f.hooks[1:]
return hook
}
func (f *UsersStoreUseCustomAvatarFunc) appendCall(r0 UsersStoreUseCustomAvatarFuncCall) {
f.mutex.Lock()
f.history = append(f.history, r0)
f.mutex.Unlock()
}
// History returns a sequence of UsersStoreUseCustomAvatarFuncCall objects
// describing the invocations of this function.
func (f *UsersStoreUseCustomAvatarFunc) History() []UsersStoreUseCustomAvatarFuncCall {
f.mutex.Lock()
history := make([]UsersStoreUseCustomAvatarFuncCall, len(f.history))
copy(history, f.history)
f.mutex.Unlock()
return history
}
// UsersStoreUseCustomAvatarFuncCall is an object that describes an
// invocation of method UseCustomAvatar on an instance of MockUsersStore.
type UsersStoreUseCustomAvatarFuncCall struct {
// Arg0 is the value of the 1st argument passed to this method
// invocation.
Arg0 context.Context
// Arg1 is the value of the 2nd argument passed to this method
// invocation.
Arg1 int64
// Arg2 is the value of the 3rd argument passed to this method
// invocation.
Arg2 []byte
// Result0 is the value of the 1st result returned from this method
// invocation.
Result0 error
}
// Args returns an interface slice containing the arguments of this
// invocation.
func (c UsersStoreUseCustomAvatarFuncCall) Args() []interface{} {
return []interface{}{c.Arg0, c.Arg1, c.Arg2}
}
// Results returns an interface slice containing the results of this
// invocation.
func (c UsersStoreUseCustomAvatarFuncCall) Results() []interface{} {
return []interface{}{c.Result0}
}

View File

@ -96,7 +96,7 @@ func SettingsAvatar(c *context.Context, f form.Avatar) {
}
func SettingsDeleteAvatar(c *context.Context) {
if err := c.Org.Organization.DeleteAvatar(); err != nil {
if err := db.Users.DeleteCustomAvatar(c.Req.Context(), c.Org.Organization.ID); err != nil {
c.Flash.Error(err.Error())
}

View File

@ -13,9 +13,9 @@ import (
"io"
"strings"
"github.com/pkg/errors"
"github.com/pquerna/otp"
"github.com/pquerna/otp/totp"
"github.com/unknwon/com"
log "unknwon.dev/clog/v2"
"gogs.io/gogs/internal/auth"
@ -23,7 +23,6 @@ import (
"gogs.io/gogs/internal/context"
"gogs.io/gogs/internal/cryptoutil"
"gogs.io/gogs/internal/db"
"gogs.io/gogs/internal/db/errors"
"gogs.io/gogs/internal/email"
"gogs.io/gogs/internal/form"
"gogs.io/gogs/internal/tool"
@ -117,10 +116,15 @@ func SettingsPost(c *context.Context, f form.UpdateProfile) {
// FIXME: limit upload size
func UpdateAvatarSetting(c *context.Context, f form.Avatar, ctxUser *db.User) error {
ctxUser.UseCustomAvatar = f.Source == form.AVATAR_LOCAL
if len(f.Gravatar) > 0 {
if f.Source == form.AVATAR_BYMAIL && len(f.Gravatar) > 0 {
ctxUser.UseCustomAvatar = false
ctxUser.Avatar = cryptoutil.MD5(f.Gravatar)
ctxUser.AvatarEmail = f.Gravatar
if err := db.UpdateUser(ctxUser); err != nil {
return fmt.Errorf("update user: %v", err)
}
return nil
}
if f.Avatar != nil && f.Avatar.Filename != "" {
@ -128,9 +132,7 @@ func UpdateAvatarSetting(c *context.Context, f form.Avatar, ctxUser *db.User) er
if err != nil {
return fmt.Errorf("open avatar reader: %v", err)
}
defer func() {
_ = r.Close()
}()
defer func() { _ = r.Close() }()
data, err := io.ReadAll(r)
if err != nil {
@ -139,23 +141,13 @@ func UpdateAvatarSetting(c *context.Context, f form.Avatar, ctxUser *db.User) er
if !tool.IsImageFile(data) {
return errors.New(c.Tr("settings.uploaded_avatar_not_a_image"))
}
if err = ctxUser.UploadAvatar(data); err != nil {
return fmt.Errorf("upload avatar: %v", err)
}
} else {
// No avatar is uploaded but setting has been changed to enable,
// generate a random one when needed.
if ctxUser.UseCustomAvatar && !com.IsFile(userutil.CustomAvatarPath(ctxUser.ID)) {
if err := userutil.GenerateRandomAvatar(ctxUser.ID, ctxUser.Name, ctxUser.Email); err != nil {
log.Error("generate random avatar [%d]: %v", ctxUser.ID, err)
}
}
}
if err := db.UpdateUser(ctxUser); err != nil {
return fmt.Errorf("update user: %v", err)
err = db.Users.UseCustomAvatar(c.Req.Context(), ctxUser.ID, data)
if err != nil {
return errors.Wrap(err, "save avatar")
}
return nil
}
return nil
}
@ -176,7 +168,8 @@ func SettingsAvatarPost(c *context.Context, f form.Avatar) {
}
func SettingsDeleteAvatar(c *context.Context) {
if err := c.User.DeleteAvatar(); err != nil {
err := db.Users.DeleteCustomAvatar(c.Req.Context(), c.User.ID)
if err != nil {
c.Flash.Error(fmt.Sprintf("Failed to delete avatar: %v", err))
}

View File

@ -5,16 +5,19 @@
package userutil
import (
"bytes"
"crypto/sha256"
"crypto/subtle"
"encoding/hex"
"fmt"
"image"
"image/png"
"os"
"path/filepath"
"strconv"
"strings"
"github.com/nfnt/resize"
"github.com/pkg/errors"
"golang.org/x/crypto/pbkdf2"
@ -81,6 +84,32 @@ func GenerateRandomAvatar(userID int64, name, email string) error {
return nil
}
// SaveAvatar saves the given avatar for the user.
func SaveAvatar(userID int64, data []byte) error {
img, _, err := image.Decode(bytes.NewReader(data))
if err != nil {
return errors.Wrap(err, "decode image")
}
avatarPath := CustomAvatarPath(userID)
err = os.MkdirAll(filepath.Dir(avatarPath), os.ModePerm)
if err != nil {
return errors.Wrap(err, "create avatar directory")
}
f, err := os.Create(avatarPath)
if err != nil {
return errors.Wrap(err, "create avatar file")
}
defer func() { _ = f.Close() }()
m := resize.Resize(avatar.DefaultSize, avatar.DefaultSize, img, resize.NearestNeighbor)
if err = png.Encode(f, m); err != nil {
return errors.Wrap(err, "encode avatar image to file")
}
return nil
}
// EncodePassword encodes password using PBKDF2 SHA256 with given salt.
func EncodePassword(password, salt string) string {
newPasswd := pbkdf2.Key([]byte(password), []byte(salt), 10000, 50, sha256.New)

View File

@ -15,6 +15,7 @@ import (
"gogs.io/gogs/internal/conf"
"gogs.io/gogs/internal/osutil"
"gogs.io/gogs/internal/tool"
"gogs.io/gogs/public"
)
func TestDashboardURLPath(t *testing.T) {
@ -72,9 +73,36 @@ func TestGenerateRandomAvatar(t *testing.T) {
},
)
avatarPath := CustomAvatarPath(1)
defer func() { _ = os.Remove(avatarPath) }()
err := GenerateRandomAvatar(1, "alice", "alice@example.com")
require.NoError(t, err)
got := osutil.IsFile(CustomAvatarPath(1))
got := osutil.IsFile(avatarPath)
assert.True(t, got)
}
func TestSaveAvatar(t *testing.T) {
if runtime.GOOS == "windows" {
t.Skip("Skipping testing on Windows")
return
}
conf.SetMockPicture(t,
conf.PictureOpts{
AvatarUploadPath: os.TempDir(),
},
)
avatar, err := public.Files.ReadFile("img/avatar_default.png")
require.NoError(t, err)
avatarPath := CustomAvatarPath(1)
defer func() { _ = os.Remove(avatarPath) }()
err = SaveAvatar(1, avatar)
require.NoError(t, err)
got := osutil.IsFile(avatarPath)
assert.True(t, got)
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 443 KiB

After

Width:  |  Height:  |  Size: 167 KiB