* Avatar support minio * Support repo avatar minio storage * Add missing migration * Fix bug * Fix test * Add test for minio store type on avatars and repo avatars; Add documents * Fix bug * Fix bug * Add back missed avatar link method * refactor codes * Simplify the codes * Code improvements * Fix lint * Fix test mysql * Fix test mysql * Fix test mysql * Fix settings * Fix test * fix test * Fix bugtags/v1.13.0-rc1
@@ -91,6 +91,20 @@ func migrateLFS(dstStorage storage.ObjectStorage) error { | |||||
}) | }) | ||||
} | } | ||||
func migrateAvatars(dstStorage storage.ObjectStorage) error { | |||||
return models.IterateUser(func(user *models.User) error { | |||||
_, err := storage.Copy(dstStorage, user.CustomAvatarRelativePath(), storage.Avatars, user.CustomAvatarRelativePath()) | |||||
return err | |||||
}) | |||||
} | |||||
func migrateRepoAvatars(dstStorage storage.ObjectStorage) error { | |||||
return models.IterateRepository(func(repo *models.Repository) error { | |||||
_, err := storage.Copy(dstStorage, repo.CustomAvatarRelativePath(), storage.RepoAvatars, repo.CustomAvatarRelativePath()) | |||||
return err | |||||
}) | |||||
} | |||||
func runMigrateStorage(ctx *cli.Context) error { | func runMigrateStorage(ctx *cli.Context) error { | ||||
if err := initDB(); err != nil { | if err := initDB(); err != nil { | ||||
return err | return err | ||||
@@ -142,9 +156,8 @@ func runMigrateStorage(ctx *cli.Context) error { | |||||
UseSSL: ctx.Bool("minio-use-ssl"), | UseSSL: ctx.Bool("minio-use-ssl"), | ||||
}) | }) | ||||
default: | default: | ||||
return fmt.Errorf("Unsupported attachments storage type: %s", ctx.String("storage")) | |||||
return fmt.Errorf("Unsupported storage type: %s", ctx.String("storage")) | |||||
} | } | ||||
if err != nil { | if err != nil { | ||||
return err | return err | ||||
} | } | ||||
@@ -159,6 +172,14 @@ func runMigrateStorage(ctx *cli.Context) error { | |||||
if err := migrateLFS(dstStorage); err != nil { | if err := migrateLFS(dstStorage); err != nil { | ||||
return err | return err | ||||
} | } | ||||
case "avatars": | |||||
if err := migrateAvatars(dstStorage); err != nil { | |||||
return err | |||||
} | |||||
case "repo-avatars": | |||||
if err := migrateRepoAvatars(dstStorage); err != nil { | |||||
return err | |||||
} | |||||
default: | default: | ||||
return fmt.Errorf("Unsupported storage: %s", ctx.String("type")) | return fmt.Errorf("Unsupported storage: %s", ctx.String("type")) | ||||
} | } | ||||
@@ -564,16 +564,21 @@ Define allowed algorithms and their minimum key length (use -1 to disable a type | |||||
- `DISABLE_GRAVATAR`: **false**: Enable this to use local avatars only. | - `DISABLE_GRAVATAR`: **false**: Enable this to use local avatars only. | ||||
- `ENABLE_FEDERATED_AVATAR`: **false**: Enable support for federated avatars (see | - `ENABLE_FEDERATED_AVATAR`: **false**: Enable support for federated avatars (see | ||||
[http://www.libravatar.org](http://www.libravatar.org)). | [http://www.libravatar.org](http://www.libravatar.org)). | ||||
- `AVATAR_STORAGE_TYPE`: **default**: Storage type defined in `[storage.xxx]`. Default is `default` which will read `[storage]` if no section `[storage]` will be a type `local`. | |||||
- `AVATAR_UPLOAD_PATH`: **data/avatars**: Path to store user avatar image files. | - `AVATAR_UPLOAD_PATH`: **data/avatars**: Path to store user avatar image files. | ||||
- `AVATAR_MAX_WIDTH`: **4096**: Maximum avatar image width in pixels. | |||||
- `AVATAR_MAX_HEIGHT`: **3072**: Maximum avatar image height in pixels. | |||||
- `AVATAR_MAX_FILE_SIZE`: **1048576** (1Mb): Maximum avatar image file size in bytes. | |||||
- `REPOSITORY_AVATAR_STORAGE_TYPE`: **default**: Storage type defined in `[storage.xxx]`. Default is `default` which will read `[storage]` if no section `[storage]` will be a type `local`. | |||||
- `REPOSITORY_AVATAR_UPLOAD_PATH`: **data/repo-avatars**: Path to store repository avatar image files. | - `REPOSITORY_AVATAR_UPLOAD_PATH`: **data/repo-avatars**: Path to store repository avatar image files. | ||||
- `REPOSITORY_AVATAR_FALLBACK`: **none**: How Gitea deals with missing repository avatars | - `REPOSITORY_AVATAR_FALLBACK`: **none**: How Gitea deals with missing repository avatars | ||||
- none = no avatar will be displayed | - none = no avatar will be displayed | ||||
- random = random avatar will be generated | - random = random avatar will be generated | ||||
- image = default image will be used (which is set in `REPOSITORY_AVATAR_DEFAULT_IMAGE`) | |||||
- image = default image will be used (which is set in `REPOSITORY_AVATAR_FALLBACK_IMAGE`) | |||||
- `REPOSITORY_AVATAR_FALLBACK_IMAGE`: **/img/repo_default.png**: Image used as default repository avatar (if `REPOSITORY_AVATAR_FALLBACK` is set to image and none was uploaded) | - `REPOSITORY_AVATAR_FALLBACK_IMAGE`: **/img/repo_default.png**: Image used as default repository avatar (if `REPOSITORY_AVATAR_FALLBACK` is set to image and none was uploaded) | ||||
- `AVATAR_MAX_WIDTH`: **4096**: Maximum avatar image width in pixels. | |||||
- `AVATAR_MAX_HEIGHT`: **3072**: Maximum avatar image height in pixels. | |||||
- `AVATAR_MAX_FILE_SIZE`: **1048576** (1Mb): Maximum avatar image file size in bytes. | |||||
## Project (`project`) | ## Project (`project`) | ||||
@@ -182,6 +182,20 @@ menu: | |||||
- `DISABLE_GRAVATAR`: 开启则只使用内部头像。 | - `DISABLE_GRAVATAR`: 开启则只使用内部头像。 | ||||
- `ENABLE_FEDERATED_AVATAR`: 启用头像联盟支持 (参见 http://www.libravatar.org) | - `ENABLE_FEDERATED_AVATAR`: 启用头像联盟支持 (参见 http://www.libravatar.org) | ||||
- `AVATAR_STORAGE_TYPE`: **local**: 头像存储类型,可以为 `local` 或 `minio`,分别支持本地文件系统和 minio 兼容的API。 | |||||
- `AVATAR_UPLOAD_PATH`: **data/avatars**: 存储头像的文件系统路径。 | |||||
- `AVATAR_MAX_WIDTH`: **4096**: 头像最大宽度,单位像素。 | |||||
- `AVATAR_MAX_HEIGHT`: **3072**: 头像最大高度,单位像素。 | |||||
- `AVATAR_MAX_FILE_SIZE`: **1048576** (1Mb): 头像最大大小。 | |||||
- `REPOSITORY_AVATAR_STORAGE_TYPE`: **local**: 仓库头像存储类型,可以为 `local` 或 `minio`,分别支持本地文件系统和 minio 兼容的API。 | |||||
- `REPOSITORY_AVATAR_UPLOAD_PATH`: **data/repo-avatars**: 存储仓库头像的路径。 | |||||
- `REPOSITORY_AVATAR_FALLBACK`: **none**: 当头像丢失时的处理方式 | |||||
- none = 不显示头像 | |||||
- random = 显示随机生成的头像 | |||||
- image = 显示默认头像,通过 `REPOSITORY_AVATAR_FALLBACK_IMAGE` 设置 | |||||
- `REPOSITORY_AVATAR_FALLBACK_IMAGE`: **/img/repo_default.png**: 默认仓库头像 | |||||
## Attachment (`attachment`) | ## Attachment (`attachment`) | ||||
- `ENABLED`: 是否允许用户上传附件。 | - `ENABLED`: 是否允许用户上传附件。 | ||||
@@ -58,7 +58,7 @@ LFS_MINIO_BASE_PATH = lfs/ | |||||
LFS_MINIO_USE_SSL = false | LFS_MINIO_USE_SSL = false | ||||
[attachment] | [attachment] | ||||
STORE_TYPE = minio | |||||
STORAGE_TYPE = minio | |||||
SERVE_DIRECT = false | SERVE_DIRECT = false | ||||
MINIO_ENDPOINT = minio:9000 | MINIO_ENDPOINT = minio:9000 | ||||
MINIO_ACCESS_KEY_ID = 123456 | MINIO_ACCESS_KEY_ID = 123456 | ||||
@@ -87,6 +87,7 @@ ENABLE_NOTIFY_MAIL = true | |||||
[picture] | [picture] | ||||
DISABLE_GRAVATAR = false | DISABLE_GRAVATAR = false | ||||
ENABLE_FEDERATED_AVATAR = false | ENABLE_FEDERATED_AVATAR = false | ||||
AVATAR_UPLOAD_PATH = integrations/gitea-integration-mysql/data/avatars | AVATAR_UPLOAD_PATH = integrations/gitea-integration-mysql/data/avatars | ||||
REPOSITORY_AVATAR_UPLOAD_PATH = integrations/gitea-integration-mysql/data/repo-avatars | REPOSITORY_AVATAR_UPLOAD_PATH = integrations/gitea-integration-mysql/data/repo-avatars | ||||
@@ -61,7 +61,7 @@ func renameExistingUserAvatarName(x *xorm.Engine) error { | |||||
for _, user := range users { | for _, user := range users { | ||||
oldAvatar := user.Avatar | oldAvatar := user.Avatar | ||||
if stat, err := os.Stat(filepath.Join(setting.AvatarUploadPath, oldAvatar)); err != nil || !stat.Mode().IsRegular() { | |||||
if stat, err := os.Stat(filepath.Join(setting.Avatar.Path, oldAvatar)); err != nil || !stat.Mode().IsRegular() { | |||||
if err == nil { | if err == nil { | ||||
err = fmt.Errorf("Error: \"%s\" is not a regular file", oldAvatar) | err = fmt.Errorf("Error: \"%s\" is not a regular file", oldAvatar) | ||||
} | } | ||||
@@ -86,7 +86,7 @@ func renameExistingUserAvatarName(x *xorm.Engine) error { | |||||
return fmt.Errorf("[user: %s] user table update: %v", user.LowerName, err) | return fmt.Errorf("[user: %s] user table update: %v", user.LowerName, err) | ||||
} | } | ||||
deleteList[filepath.Join(setting.AvatarUploadPath, oldAvatar)] = struct{}{} | |||||
deleteList[filepath.Join(setting.Avatar.Path, oldAvatar)] = struct{}{} | |||||
migrated++ | migrated++ | ||||
select { | select { | ||||
case <-ticker.C: | case <-ticker.C: | ||||
@@ -135,7 +135,7 @@ func renameExistingUserAvatarName(x *xorm.Engine) error { | |||||
// copyOldAvatarToNewLocation copies oldAvatar to newAvatarLocation | // copyOldAvatarToNewLocation copies oldAvatar to newAvatarLocation | ||||
// and returns newAvatar location | // and returns newAvatar location | ||||
func copyOldAvatarToNewLocation(userID int64, oldAvatar string) (string, error) { | func copyOldAvatarToNewLocation(userID int64, oldAvatar string) (string, error) { | ||||
fr, err := os.Open(filepath.Join(setting.AvatarUploadPath, oldAvatar)) | |||||
fr, err := os.Open(filepath.Join(setting.Avatar.Path, oldAvatar)) | |||||
if err != nil { | if err != nil { | ||||
return "", fmt.Errorf("os.Open: %v", err) | return "", fmt.Errorf("os.Open: %v", err) | ||||
} | } | ||||
@@ -151,7 +151,7 @@ func copyOldAvatarToNewLocation(userID int64, oldAvatar string) (string, error) | |||||
return newAvatar, nil | return newAvatar, nil | ||||
} | } | ||||
if err := ioutil.WriteFile(filepath.Join(setting.AvatarUploadPath, newAvatar), data, 0666); err != nil { | |||||
if err := ioutil.WriteFile(filepath.Join(setting.Avatar.Path, newAvatar), data, 0666); err != nil { | |||||
return "", fmt.Errorf("ioutil.WriteFile: %v", err) | return "", fmt.Errorf("ioutil.WriteFile: %v", err) | ||||
} | } | ||||
@@ -11,10 +11,10 @@ import ( | |||||
"code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
"code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
"code.gitea.io/gitea/modules/storage" | |||||
"code.gitea.io/gitea/modules/structs" | "code.gitea.io/gitea/modules/structs" | ||||
"code.gitea.io/gitea/modules/util" | "code.gitea.io/gitea/modules/util" | ||||
"github.com/unknwon/com" | |||||
"xorm.io/builder" | "xorm.io/builder" | ||||
"xorm.io/xorm" | "xorm.io/xorm" | ||||
) | ) | ||||
@@ -310,11 +310,9 @@ func deleteOrg(e *xorm.Session, u *User) error { | |||||
} | } | ||||
if len(u.Avatar) > 0 { | if len(u.Avatar) > 0 { | ||||
avatarPath := u.CustomAvatarPath() | |||||
if com.IsExist(avatarPath) { | |||||
if err := util.Remove(avatarPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", avatarPath, err) | |||||
} | |||||
avatarPath := u.CustomAvatarRelativePath() | |||||
if err := storage.Avatars.Delete(avatarPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", avatarPath, err) | |||||
} | } | ||||
} | } | ||||
@@ -7,7 +7,6 @@ package models | |||||
import ( | import ( | ||||
"context" | "context" | ||||
"crypto/md5" | |||||
"errors" | "errors" | ||||
"fmt" | "fmt" | ||||
"html/template" | "html/template" | ||||
@@ -15,7 +14,6 @@ import ( | |||||
// Needed for jpeg support | // Needed for jpeg support | ||||
_ "image/jpeg" | _ "image/jpeg" | ||||
"image/png" | |||||
"io/ioutil" | "io/ioutil" | ||||
"net" | "net" | ||||
"net/url" | "net/url" | ||||
@@ -27,7 +25,6 @@ import ( | |||||
"strings" | "strings" | ||||
"time" | "time" | ||||
"code.gitea.io/gitea/modules/avatar" | |||||
"code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
"code.gitea.io/gitea/modules/markup" | "code.gitea.io/gitea/modules/markup" | ||||
"code.gitea.io/gitea/modules/options" | "code.gitea.io/gitea/modules/options" | ||||
@@ -1796,11 +1793,8 @@ func DeleteRepository(doer *User, uid, repoID int64) error { | |||||
} | } | ||||
if len(repo.Avatar) > 0 { | if len(repo.Avatar) > 0 { | ||||
avatarPath := repo.CustomAvatarPath() | |||||
if com.IsExist(avatarPath) { | |||||
if err := util.Remove(avatarPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", avatarPath, err) | |||||
} | |||||
if err := storage.RepoAvatars.Delete(repo.CustomAvatarRelativePath()); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", repo.Avatar, err) | |||||
} | } | ||||
} | } | ||||
@@ -2239,187 +2233,6 @@ func (repo *Repository) GetUserFork(userID int64) (*Repository, error) { | |||||
return &forkedRepo, nil | return &forkedRepo, nil | ||||
} | } | ||||
// CustomAvatarPath returns repository custom avatar file path. | |||||
func (repo *Repository) CustomAvatarPath() string { | |||||
// Avatar empty by default | |||||
if len(repo.Avatar) == 0 { | |||||
return "" | |||||
} | |||||
return filepath.Join(setting.RepositoryAvatarUploadPath, repo.Avatar) | |||||
} | |||||
// generateRandomAvatar generates a random avatar for repository. | |||||
func (repo *Repository) generateRandomAvatar(e Engine) error { | |||||
idToString := fmt.Sprintf("%d", repo.ID) | |||||
seed := idToString | |||||
img, err := avatar.RandomImage([]byte(seed)) | |||||
if err != nil { | |||||
return fmt.Errorf("RandomImage: %v", err) | |||||
} | |||||
repo.Avatar = idToString | |||||
if err = os.MkdirAll(filepath.Dir(repo.CustomAvatarPath()), os.ModePerm); err != nil { | |||||
return fmt.Errorf("MkdirAll: %v", err) | |||||
} | |||||
fw, err := os.Create(repo.CustomAvatarPath()) | |||||
if err != nil { | |||||
return fmt.Errorf("Create: %v", err) | |||||
} | |||||
defer fw.Close() | |||||
if err = png.Encode(fw, img); err != nil { | |||||
return fmt.Errorf("Encode: %v", err) | |||||
} | |||||
log.Info("New random avatar created for repository: %d", repo.ID) | |||||
if _, err := e.ID(repo.ID).Cols("avatar").NoAutoTime().Update(repo); err != nil { | |||||
return err | |||||
} | |||||
return nil | |||||
} | |||||
// RemoveRandomAvatars removes the randomly generated avatars that were created for repositories | |||||
func RemoveRandomAvatars(ctx context.Context) error { | |||||
return x. | |||||
Where("id > 0").BufferSize(setting.Database.IterateBufferSize). | |||||
Iterate(new(Repository), | |||||
func(idx int, bean interface{}) error { | |||||
repository := bean.(*Repository) | |||||
select { | |||||
case <-ctx.Done(): | |||||
return ErrCancelledf("before random avatars removed for %s", repository.FullName()) | |||||
default: | |||||
} | |||||
stringifiedID := strconv.FormatInt(repository.ID, 10) | |||||
if repository.Avatar == stringifiedID { | |||||
return repository.DeleteAvatar() | |||||
} | |||||
return nil | |||||
}) | |||||
} | |||||
// RelAvatarLink returns a relative link to the repository's avatar. | |||||
func (repo *Repository) RelAvatarLink() string { | |||||
return repo.relAvatarLink(x) | |||||
} | |||||
func (repo *Repository) relAvatarLink(e Engine) string { | |||||
// If no avatar - path is empty | |||||
avatarPath := repo.CustomAvatarPath() | |||||
if len(avatarPath) == 0 || !com.IsFile(avatarPath) { | |||||
switch mode := setting.RepositoryAvatarFallback; mode { | |||||
case "image": | |||||
return setting.RepositoryAvatarFallbackImage | |||||
case "random": | |||||
if err := repo.generateRandomAvatar(e); err != nil { | |||||
log.Error("generateRandomAvatar: %v", err) | |||||
} | |||||
default: | |||||
// default behaviour: do not display avatar | |||||
return "" | |||||
} | |||||
} | |||||
return setting.AppSubURL + "/repo-avatars/" + repo.Avatar | |||||
} | |||||
// AvatarLink returns a link to the repository's avatar. | |||||
func (repo *Repository) AvatarLink() string { | |||||
return repo.avatarLink(x) | |||||
} | |||||
// avatarLink returns user avatar absolute link. | |||||
func (repo *Repository) avatarLink(e Engine) string { | |||||
link := repo.relAvatarLink(e) | |||||
// link may be empty! | |||||
if len(link) > 0 { | |||||
if link[0] == '/' && link[1] != '/' { | |||||
return setting.AppURL + strings.TrimPrefix(link, setting.AppSubURL)[1:] | |||||
} | |||||
} | |||||
return link | |||||
} | |||||
// UploadAvatar saves custom avatar for repository. | |||||
// FIXME: split uploads to different subdirs in case we have massive number of repos. | |||||
func (repo *Repository) UploadAvatar(data []byte) error { | |||||
m, err := avatar.Prepare(data) | |||||
if err != nil { | |||||
return err | |||||
} | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err = sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
oldAvatarPath := repo.CustomAvatarPath() | |||||
// Users can upload the same image to other repo - prefix it with ID | |||||
// Then repo will be removed - only it avatar file will be removed | |||||
repo.Avatar = fmt.Sprintf("%d-%x", repo.ID, md5.Sum(data)) | |||||
if _, err := sess.ID(repo.ID).Cols("avatar").Update(repo); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Update repository avatar: %v", err) | |||||
} | |||||
if err := os.MkdirAll(setting.RepositoryAvatarUploadPath, os.ModePerm); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Failed to create dir %s: %v", setting.RepositoryAvatarUploadPath, err) | |||||
} | |||||
fw, err := os.Create(repo.CustomAvatarPath()) | |||||
if err != nil { | |||||
return fmt.Errorf("UploadAvatar: Create file: %v", err) | |||||
} | |||||
defer fw.Close() | |||||
if err = png.Encode(fw, *m); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Encode png: %v", err) | |||||
} | |||||
if len(oldAvatarPath) > 0 && oldAvatarPath != repo.CustomAvatarPath() { | |||||
if err := util.Remove(oldAvatarPath); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Failed to remove old repo avatar %s: %v", oldAvatarPath, err) | |||||
} | |||||
} | |||||
return sess.Commit() | |||||
} | |||||
// DeleteAvatar deletes the repos's custom avatar. | |||||
func (repo *Repository) DeleteAvatar() error { | |||||
// Avatar not exists | |||||
if len(repo.Avatar) == 0 { | |||||
return nil | |||||
} | |||||
avatarPath := repo.CustomAvatarPath() | |||||
log.Trace("DeleteAvatar[%d]: %s", repo.ID, avatarPath) | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err := sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
repo.Avatar = "" | |||||
if _, err := sess.ID(repo.ID).Cols("avatar").Update(repo); err != nil { | |||||
return fmt.Errorf("DeleteAvatar: Update repository avatar: %v", err) | |||||
} | |||||
if _, err := os.Stat(avatarPath); err == nil { | |||||
if err := util.Remove(avatarPath); err != nil { | |||||
return fmt.Errorf("DeleteAvatar: Failed to remove %s: %v", avatarPath, err) | |||||
} | |||||
} else { | |||||
// // Schrodinger: file may or may not exist. See err for details. | |||||
log.Trace("DeleteAvatar[%d]: %v", err) | |||||
} | |||||
return sess.Commit() | |||||
} | |||||
// GetOriginalURLHostname returns the hostname of a URL or the URL | // GetOriginalURLHostname returns the hostname of a URL or the URL | ||||
func (repo *Repository) GetOriginalURLHostname() string { | func (repo *Repository) GetOriginalURLHostname() string { | ||||
u, err := url.Parse(repo.OriginalURL) | u, err := url.Parse(repo.OriginalURL) | ||||
@@ -2502,3 +2315,25 @@ func DoctorUserStarNum() (err error) { | |||||
return | return | ||||
} | } | ||||
// IterateRepository iterate repositories | |||||
func IterateRepository(f func(repo *Repository) error) error { | |||||
var start int | |||||
var batchSize = setting.Database.IterateBufferSize | |||||
for { | |||||
var repos = make([]*Repository, 0, batchSize) | |||||
if err := x.Limit(batchSize, start).Find(&repos); err != nil { | |||||
return err | |||||
} | |||||
if len(repos) == 0 { | |||||
return nil | |||||
} | |||||
start += len(repos) | |||||
for _, repo := range repos { | |||||
if err := f(repo); err != nil { | |||||
return err | |||||
} | |||||
} | |||||
} | |||||
} |
@@ -0,0 +1,190 @@ | |||||
// Copyright 2020 The Gitea 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 models | |||||
import ( | |||||
"context" | |||||
"crypto/md5" | |||||
"fmt" | |||||
"image/png" | |||||
"io" | |||||
"strconv" | |||||
"strings" | |||||
"code.gitea.io/gitea/modules/avatar" | |||||
"code.gitea.io/gitea/modules/log" | |||||
"code.gitea.io/gitea/modules/setting" | |||||
"code.gitea.io/gitea/modules/storage" | |||||
) | |||||
// CustomAvatarRelativePath returns repository custom avatar file path. | |||||
func (repo *Repository) CustomAvatarRelativePath() string { | |||||
return repo.Avatar | |||||
} | |||||
// generateRandomAvatar generates a random avatar for repository. | |||||
func (repo *Repository) generateRandomAvatar(e Engine) error { | |||||
idToString := fmt.Sprintf("%d", repo.ID) | |||||
seed := idToString | |||||
img, err := avatar.RandomImage([]byte(seed)) | |||||
if err != nil { | |||||
return fmt.Errorf("RandomImage: %v", err) | |||||
} | |||||
repo.Avatar = idToString | |||||
if err := storage.SaveFrom(storage.RepoAvatars, repo.CustomAvatarRelativePath(), func(w io.Writer) error { | |||||
if err := png.Encode(w, img); err != nil { | |||||
log.Error("Encode: %v", err) | |||||
} | |||||
return err | |||||
}); err != nil { | |||||
return fmt.Errorf("Failed to create dir %s: %v", repo.CustomAvatarRelativePath(), err) | |||||
} | |||||
log.Info("New random avatar created for repository: %d", repo.ID) | |||||
if _, err := e.ID(repo.ID).Cols("avatar").NoAutoTime().Update(repo); err != nil { | |||||
return err | |||||
} | |||||
return nil | |||||
} | |||||
// RemoveRandomAvatars removes the randomly generated avatars that were created for repositories | |||||
func RemoveRandomAvatars(ctx context.Context) error { | |||||
return x. | |||||
Where("id > 0").BufferSize(setting.Database.IterateBufferSize). | |||||
Iterate(new(Repository), | |||||
func(idx int, bean interface{}) error { | |||||
repository := bean.(*Repository) | |||||
select { | |||||
case <-ctx.Done(): | |||||
return ErrCancelledf("before random avatars removed for %s", repository.FullName()) | |||||
default: | |||||
} | |||||
stringifiedID := strconv.FormatInt(repository.ID, 10) | |||||
if repository.Avatar == stringifiedID { | |||||
return repository.DeleteAvatar() | |||||
} | |||||
return nil | |||||
}) | |||||
} | |||||
// RelAvatarLink returns a relative link to the repository's avatar. | |||||
func (repo *Repository) RelAvatarLink() string { | |||||
return repo.relAvatarLink(x) | |||||
} | |||||
func (repo *Repository) relAvatarLink(e Engine) string { | |||||
// If no avatar - path is empty | |||||
avatarPath := repo.CustomAvatarRelativePath() | |||||
if len(avatarPath) == 0 { | |||||
switch mode := setting.RepoAvatar.Fallback; mode { | |||||
case "image": | |||||
return setting.RepoAvatar.FallbackImage | |||||
case "random": | |||||
if err := repo.generateRandomAvatar(e); err != nil { | |||||
log.Error("generateRandomAvatar: %v", err) | |||||
} | |||||
default: | |||||
// default behaviour: do not display avatar | |||||
return "" | |||||
} | |||||
} | |||||
return setting.AppSubURL + "/repo-avatars/" + repo.Avatar | |||||
} | |||||
// AvatarLink returns a link to the repository's avatar. | |||||
func (repo *Repository) AvatarLink() string { | |||||
return repo.avatarLink(x) | |||||
} | |||||
// avatarLink returns user avatar absolute link. | |||||
func (repo *Repository) avatarLink(e Engine) string { | |||||
link := repo.relAvatarLink(e) | |||||
// link may be empty! | |||||
if len(link) > 0 { | |||||
if link[0] == '/' && link[1] != '/' { | |||||
return setting.AppURL + strings.TrimPrefix(link, setting.AppSubURL)[1:] | |||||
} | |||||
} | |||||
return link | |||||
} | |||||
// UploadAvatar saves custom avatar for repository. | |||||
// FIXME: split uploads to different subdirs in case we have massive number of repos. | |||||
func (repo *Repository) UploadAvatar(data []byte) error { | |||||
m, err := avatar.Prepare(data) | |||||
if err != nil { | |||||
return err | |||||
} | |||||
newAvatar := fmt.Sprintf("%d-%x", repo.ID, md5.Sum(data)) | |||||
if repo.Avatar == newAvatar { // upload the same picture | |||||
return nil | |||||
} | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err = sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
oldAvatarPath := repo.CustomAvatarRelativePath() | |||||
// Users can upload the same image to other repo - prefix it with ID | |||||
// Then repo will be removed - only it avatar file will be removed | |||||
repo.Avatar = newAvatar | |||||
if _, err := sess.ID(repo.ID).Cols("avatar").Update(repo); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Update repository avatar: %v", err) | |||||
} | |||||
if err := storage.SaveFrom(storage.RepoAvatars, repo.CustomAvatarRelativePath(), func(w io.Writer) error { | |||||
if err := png.Encode(w, *m); err != nil { | |||||
log.Error("Encode: %v", err) | |||||
} | |||||
return err | |||||
}); err != nil { | |||||
return fmt.Errorf("UploadAvatar %s failed: Failed to remove old repo avatar %s: %v", repo.RepoPath(), newAvatar, err) | |||||
} | |||||
if len(oldAvatarPath) > 0 { | |||||
if err := storage.RepoAvatars.Delete(oldAvatarPath); err != nil { | |||||
return fmt.Errorf("UploadAvatar: Failed to remove old repo avatar %s: %v", oldAvatarPath, err) | |||||
} | |||||
} | |||||
return sess.Commit() | |||||
} | |||||
// DeleteAvatar deletes the repos's custom avatar. | |||||
func (repo *Repository) DeleteAvatar() error { | |||||
// Avatar not exists | |||||
if len(repo.Avatar) == 0 { | |||||
return nil | |||||
} | |||||
avatarPath := repo.CustomAvatarRelativePath() | |||||
log.Trace("DeleteAvatar[%d]: %s", repo.ID, avatarPath) | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err := sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
repo.Avatar = "" | |||||
if _, err := sess.ID(repo.ID).Cols("avatar").Update(repo); err != nil { | |||||
return fmt.Errorf("DeleteAvatar: Update repository avatar: %v", err) | |||||
} | |||||
if err := storage.RepoAvatars.Delete(avatarPath); err != nil { | |||||
return fmt.Errorf("DeleteAvatar: Failed to remove %s: %v", avatarPath, err) | |||||
} | |||||
return sess.Commit() | |||||
} |
@@ -10,10 +10,10 @@ import ( | |||||
"code.gitea.io/gitea/modules/git" | "code.gitea.io/gitea/modules/git" | ||||
"code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
"code.gitea.io/gitea/modules/storage" | |||||
"code.gitea.io/gitea/modules/util" | "code.gitea.io/gitea/modules/util" | ||||
"github.com/gobwas/glob" | "github.com/gobwas/glob" | ||||
"github.com/unknwon/com" | |||||
) | ) | ||||
// GenerateRepoOptions contains the template units to generate | // GenerateRepoOptions contains the template units to generate | ||||
@@ -139,7 +139,7 @@ func GenerateWebhooks(ctx DBContext, templateRepo, generateRepo *Repository) err | |||||
// GenerateAvatar generates the avatar from a template repository | // GenerateAvatar generates the avatar from a template repository | ||||
func GenerateAvatar(ctx DBContext, templateRepo, generateRepo *Repository) error { | func GenerateAvatar(ctx DBContext, templateRepo, generateRepo *Repository) error { | ||||
generateRepo.Avatar = strings.Replace(templateRepo.Avatar, strconv.FormatInt(templateRepo.ID, 10), strconv.FormatInt(generateRepo.ID, 10), 1) | generateRepo.Avatar = strings.Replace(templateRepo.Avatar, strconv.FormatInt(templateRepo.ID, 10), strconv.FormatInt(generateRepo.ID, 10), 1) | ||||
if err := com.Copy(templateRepo.CustomAvatarPath(), generateRepo.CustomAvatarPath()); err != nil { | |||||
if _, err := storage.Copy(storage.RepoAvatars, generateRepo.CustomAvatarRelativePath(), storage.RepoAvatars, templateRepo.CustomAvatarRelativePath()); err != nil { | |||||
return err | return err | ||||
} | } | ||||
@@ -70,6 +70,11 @@ func MainTest(m *testing.M, pathToGiteaRoot string) { | |||||
setting.Attachment.Storage.Path = filepath.Join(setting.AppDataPath, "attachments") | setting.Attachment.Storage.Path = filepath.Join(setting.AppDataPath, "attachments") | ||||
setting.LFS.Storage.Path = filepath.Join(setting.AppDataPath, "lfs") | setting.LFS.Storage.Path = filepath.Join(setting.AppDataPath, "lfs") | ||||
setting.Avatar.Storage.Path = filepath.Join(setting.AppDataPath, "avatars") | |||||
setting.RepoAvatar.Storage.Path = filepath.Join(setting.AppDataPath, "repo-avatars") | |||||
if err = storage.Init(); err != nil { | if err = storage.Init(); err != nil { | ||||
fatalTestError("storage.Init: %v\n", err) | fatalTestError("storage.Init: %v\n", err) | ||||
} | } | ||||
@@ -8,29 +8,26 @@ package models | |||||
import ( | import ( | ||||
"container/list" | "container/list" | ||||
"context" | "context" | ||||
"crypto/md5" | |||||
"crypto/sha256" | "crypto/sha256" | ||||
"crypto/subtle" | "crypto/subtle" | ||||
"encoding/hex" | "encoding/hex" | ||||
"errors" | "errors" | ||||
"fmt" | "fmt" | ||||
_ "image/jpeg" // Needed for jpeg support | _ "image/jpeg" // Needed for jpeg support | ||||
"image/png" | |||||
"os" | "os" | ||||
"path/filepath" | "path/filepath" | ||||
"regexp" | "regexp" | ||||
"strconv" | |||||
"strings" | "strings" | ||||
"time" | "time" | ||||
"unicode/utf8" | "unicode/utf8" | ||||
"code.gitea.io/gitea/modules/avatar" | |||||
"code.gitea.io/gitea/modules/base" | "code.gitea.io/gitea/modules/base" | ||||
"code.gitea.io/gitea/modules/generate" | "code.gitea.io/gitea/modules/generate" | ||||
"code.gitea.io/gitea/modules/git" | "code.gitea.io/gitea/modules/git" | ||||
"code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
"code.gitea.io/gitea/modules/public" | "code.gitea.io/gitea/modules/public" | ||||
"code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
"code.gitea.io/gitea/modules/storage" | |||||
"code.gitea.io/gitea/modules/structs" | "code.gitea.io/gitea/modules/structs" | ||||
api "code.gitea.io/gitea/modules/structs" | api "code.gitea.io/gitea/modules/structs" | ||||
"code.gitea.io/gitea/modules/timeutil" | "code.gitea.io/gitea/modules/timeutil" | ||||
@@ -347,104 +344,6 @@ func (u *User) GenerateActivateCode() string { | |||||
return u.GenerateEmailActivateCode(u.Email) | return u.GenerateEmailActivateCode(u.Email) | ||||
} | } | ||||
// CustomAvatarPath returns user custom avatar file path. | |||||
func (u *User) CustomAvatarPath() string { | |||||
return filepath.Join(setting.AvatarUploadPath, u.Avatar) | |||||
} | |||||
// GenerateRandomAvatar generates a random avatar for user. | |||||
func (u *User) GenerateRandomAvatar() error { | |||||
return u.generateRandomAvatar(x) | |||||
} | |||||
func (u *User) generateRandomAvatar(e Engine) error { | |||||
seed := u.Email | |||||
if len(seed) == 0 { | |||||
seed = u.Name | |||||
} | |||||
img, err := avatar.RandomImage([]byte(seed)) | |||||
if err != nil { | |||||
return fmt.Errorf("RandomImage: %v", err) | |||||
} | |||||
// NOTICE for random avatar, it still uses id as avatar name, but custom avatar use md5 | |||||
// since random image is not a user's photo, there is no security for enumable | |||||
if u.Avatar == "" { | |||||
u.Avatar = fmt.Sprintf("%d", u.ID) | |||||
} | |||||
if err = os.MkdirAll(filepath.Dir(u.CustomAvatarPath()), os.ModePerm); err != nil { | |||||
return fmt.Errorf("MkdirAll: %v", err) | |||||
} | |||||
fw, err := os.Create(u.CustomAvatarPath()) | |||||
if err != nil { | |||||
return fmt.Errorf("Create: %v", err) | |||||
} | |||||
defer fw.Close() | |||||
if _, err := e.ID(u.ID).Cols("avatar").Update(u); err != nil { | |||||
return err | |||||
} | |||||
if err = png.Encode(fw, img); err != nil { | |||||
return fmt.Errorf("Encode: %v", err) | |||||
} | |||||
log.Info("New random avatar created: %d", u.ID) | |||||
return nil | |||||
} | |||||
// SizedRelAvatarLink returns a link to the user's avatar via | |||||
// the local explore page. Function returns immediately. | |||||
// When applicable, the link is for an avatar of the indicated size (in pixels). | |||||
func (u *User) SizedRelAvatarLink(size int) string { | |||||
return strings.TrimSuffix(setting.AppSubURL, "/") + "/user/avatar/" + u.Name + "/" + strconv.Itoa(size) | |||||
} | |||||
// RealSizedAvatarLink returns a link to the user's avatar. When | |||||
// applicable, the link is for an avatar of the indicated size (in pixels). | |||||
// | |||||
// This function make take time to return when federated avatars | |||||
// are in use, due to a DNS lookup need | |||||
// | |||||
func (u *User) RealSizedAvatarLink(size int) string { | |||||
if u.ID == -1 { | |||||
return base.DefaultAvatarLink() | |||||
} | |||||
switch { | |||||
case u.UseCustomAvatar: | |||||
if !com.IsFile(u.CustomAvatarPath()) { | |||||
return base.DefaultAvatarLink() | |||||
} | |||||
return setting.AppSubURL + "/avatars/" + u.Avatar | |||||
case setting.DisableGravatar, setting.OfflineMode: | |||||
if !com.IsFile(u.CustomAvatarPath()) { | |||||
if err := u.GenerateRandomAvatar(); err != nil { | |||||
log.Error("GenerateRandomAvatar: %v", err) | |||||
} | |||||
} | |||||
return setting.AppSubURL + "/avatars/" + u.Avatar | |||||
} | |||||
return base.SizedAvatarLink(u.AvatarEmail, size) | |||||
} | |||||
// RelAvatarLink returns a relative link to the user's avatar. The link | |||||
// may either be a sub-URL to this site, or a full URL to an external avatar | |||||
// service. | |||||
func (u *User) RelAvatarLink() string { | |||||
return u.SizedRelAvatarLink(base.DefaultAvatarSize) | |||||
} | |||||
// AvatarLink returns user avatar absolute link. | |||||
func (u *User) AvatarLink() string { | |||||
link := u.RelAvatarLink() | |||||
if link[0] == '/' && link[1] != '/' { | |||||
return setting.AppURL + strings.TrimPrefix(link, setting.AppSubURL)[1:] | |||||
} | |||||
return link | |||||
} | |||||
// GetFollowers returns range of user's followers. | // GetFollowers returns range of user's followers. | ||||
func (u *User) GetFollowers(listOptions ListOptions) ([]*User, error) { | func (u *User) GetFollowers(listOptions ListOptions) ([]*User, error) { | ||||
sess := x. | sess := x. | ||||
@@ -537,64 +436,6 @@ func (u *User) IsPasswordSet() bool { | |||||
return !u.ValidatePassword("") | return !u.ValidatePassword("") | ||||
} | } | ||||
// UploadAvatar saves custom avatar for user. | |||||
// FIXME: split uploads to different subdirs in case we have massive users. | |||||
func (u *User) UploadAvatar(data []byte) error { | |||||
m, err := avatar.Prepare(data) | |||||
if err != nil { | |||||
return err | |||||
} | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err = sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
u.UseCustomAvatar = true | |||||
// Different users can upload same image as avatar | |||||
// If we prefix it with u.ID, it will be separated | |||||
// Otherwise, if any of the users delete his avatar | |||||
// Other users will lose their avatars too. | |||||
u.Avatar = fmt.Sprintf("%x", md5.Sum([]byte(fmt.Sprintf("%d-%x", u.ID, md5.Sum(data))))) | |||||
if err = updateUser(sess, u); err != nil { | |||||
return fmt.Errorf("updateUser: %v", err) | |||||
} | |||||
if err := os.MkdirAll(setting.AvatarUploadPath, os.ModePerm); err != nil { | |||||
return fmt.Errorf("Failed to create dir %s: %v", setting.AvatarUploadPath, err) | |||||
} | |||||
fw, err := os.Create(u.CustomAvatarPath()) | |||||
if err != nil { | |||||
return fmt.Errorf("Create: %v", err) | |||||
} | |||||
defer fw.Close() | |||||
if err = png.Encode(fw, *m); err != nil { | |||||
return fmt.Errorf("Encode: %v", err) | |||||
} | |||||
return sess.Commit() | |||||
} | |||||
// DeleteAvatar deletes the user's custom avatar. | |||||
func (u *User) DeleteAvatar() error { | |||||
log.Trace("DeleteAvatar[%d]: %s", u.ID, u.CustomAvatarPath()) | |||||
if len(u.Avatar) > 0 { | |||||
if err := util.Remove(u.CustomAvatarPath()); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", u.CustomAvatarPath(), err) | |||||
} | |||||
} | |||||
u.UseCustomAvatar = false | |||||
u.Avatar = "" | |||||
if _, err := x.ID(u.ID).Cols("avatar, use_custom_avatar").Update(u); err != nil { | |||||
return fmt.Errorf("UpdateUser: %v", err) | |||||
} | |||||
return nil | |||||
} | |||||
// IsOrganization returns true if user is actually a organization. | // IsOrganization returns true if user is actually a organization. | ||||
func (u *User) IsOrganization() bool { | func (u *User) IsOrganization() bool { | ||||
return u.Type == UserTypeOrganization | return u.Type == UserTypeOrganization | ||||
@@ -1285,17 +1126,14 @@ func deleteUser(e *xorm.Session, u *User) error { | |||||
// Note: There are something just cannot be roll back, | // Note: There are something just cannot be roll back, | ||||
// so just keep error logs of those operations. | // so just keep error logs of those operations. | ||||
path := UserPath(u.Name) | path := UserPath(u.Name) | ||||
if err := util.RemoveAll(path); err != nil { | if err := util.RemoveAll(path); err != nil { | ||||
return fmt.Errorf("Failed to RemoveAll %s: %v", path, err) | return fmt.Errorf("Failed to RemoveAll %s: %v", path, err) | ||||
} | } | ||||
if len(u.Avatar) > 0 { | if len(u.Avatar) > 0 { | ||||
avatarPath := u.CustomAvatarPath() | |||||
if com.IsExist(avatarPath) { | |||||
if err := util.Remove(avatarPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", avatarPath, err) | |||||
} | |||||
avatarPath := u.CustomAvatarRelativePath() | |||||
if err := storage.Avatars.Delete(avatarPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", avatarPath, err) | |||||
} | } | ||||
} | } | ||||
@@ -2034,3 +1872,25 @@ func SyncExternalUsers(ctx context.Context, updateExisting bool) error { | |||||
} | } | ||||
return nil | return nil | ||||
} | } | ||||
// IterateUser iterate users | |||||
func IterateUser(f func(user *User) error) error { | |||||
var start int | |||||
var batchSize = setting.Database.IterateBufferSize | |||||
for { | |||||
var users = make([]*User, 0, batchSize) | |||||
if err := x.Limit(batchSize, start).Find(&users); err != nil { | |||||
return err | |||||
} | |||||
if len(users) == 0 { | |||||
return nil | |||||
} | |||||
start += len(users) | |||||
for _, user := range users { | |||||
if err := f(user); err != nil { | |||||
return err | |||||
} | |||||
} | |||||
} | |||||
} |
@@ -0,0 +1,169 @@ | |||||
// Copyright 2020 The Gitea 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 models | |||||
import ( | |||||
"crypto/md5" | |||||
"fmt" | |||||
"image/png" | |||||
"io" | |||||
"strconv" | |||||
"strings" | |||||
"code.gitea.io/gitea/modules/avatar" | |||||
"code.gitea.io/gitea/modules/base" | |||||
"code.gitea.io/gitea/modules/log" | |||||
"code.gitea.io/gitea/modules/setting" | |||||
"code.gitea.io/gitea/modules/storage" | |||||
) | |||||
// CustomAvatarRelativePath returns user custom avatar relative path. | |||||
func (u *User) CustomAvatarRelativePath() string { | |||||
return u.Avatar | |||||
} | |||||
// GenerateRandomAvatar generates a random avatar for user. | |||||
func (u *User) GenerateRandomAvatar() error { | |||||
return u.generateRandomAvatar(x) | |||||
} | |||||
func (u *User) generateRandomAvatar(e Engine) error { | |||||
seed := u.Email | |||||
if len(seed) == 0 { | |||||
seed = u.Name | |||||
} | |||||
img, err := avatar.RandomImage([]byte(seed)) | |||||
if err != nil { | |||||
return fmt.Errorf("RandomImage: %v", err) | |||||
} | |||||
// NOTICE for random avatar, it still uses id as avatar name, but custom avatar use md5 | |||||
// since random image is not a user's photo, there is no security for enumable | |||||
if u.Avatar == "" { | |||||
u.Avatar = fmt.Sprintf("%d", u.ID) | |||||
} | |||||
if err := storage.SaveFrom(storage.Avatars, u.CustomAvatarRelativePath(), func(w io.Writer) error { | |||||
if err := png.Encode(w, img); err != nil { | |||||
log.Error("Encode: %v", err) | |||||
} | |||||
return err | |||||
}); err != nil { | |||||
return fmt.Errorf("Failed to create dir %s: %v", u.CustomAvatarRelativePath(), err) | |||||
} | |||||
if _, err := e.ID(u.ID).Cols("avatar").Update(u); err != nil { | |||||
return err | |||||
} | |||||
log.Info("New random avatar created: %d", u.ID) | |||||
return nil | |||||
} | |||||
// SizedRelAvatarLink returns a link to the user's avatar via | |||||
// the local explore page. Function returns immediately. | |||||
// When applicable, the link is for an avatar of the indicated size (in pixels). | |||||
func (u *User) SizedRelAvatarLink(size int) string { | |||||
return strings.TrimSuffix(setting.AppSubURL, "/") + "/user/avatar/" + u.Name + "/" + strconv.Itoa(size) | |||||
} | |||||
// RealSizedAvatarLink returns a link to the user's avatar. When | |||||
// applicable, the link is for an avatar of the indicated size (in pixels). | |||||
// | |||||
// This function make take time to return when federated avatars | |||||
// are in use, due to a DNS lookup need | |||||
// | |||||
func (u *User) RealSizedAvatarLink(size int) string { | |||||
if u.ID == -1 { | |||||
return base.DefaultAvatarLink() | |||||
} | |||||
switch { | |||||
case u.UseCustomAvatar: | |||||
if u.Avatar == "" { | |||||
return base.DefaultAvatarLink() | |||||
} | |||||
return setting.AppSubURL + "/avatars/" + u.Avatar | |||||
case setting.DisableGravatar, setting.OfflineMode: | |||||
if u.Avatar == "" { | |||||
if err := u.GenerateRandomAvatar(); err != nil { | |||||
log.Error("GenerateRandomAvatar: %v", err) | |||||
} | |||||
} | |||||
return setting.AppSubURL + "/avatars/" + u.Avatar | |||||
} | |||||
return base.SizedAvatarLink(u.AvatarEmail, size) | |||||
} | |||||
// RelAvatarLink returns a relative link to the user's avatar. The link | |||||
// may either be a sub-URL to this site, or a full URL to an external avatar | |||||
// service. | |||||
func (u *User) RelAvatarLink() string { | |||||
return u.SizedRelAvatarLink(base.DefaultAvatarSize) | |||||
} | |||||
// AvatarLink returns user avatar absolute link. | |||||
func (u *User) AvatarLink() string { | |||||
link := u.RelAvatarLink() | |||||
if link[0] == '/' && link[1] != '/' { | |||||
return setting.AppURL + strings.TrimPrefix(link, setting.AppSubURL)[1:] | |||||
} | |||||
return link | |||||
} | |||||
// UploadAvatar saves custom avatar for user. | |||||
// FIXME: split uploads to different subdirs in case we have massive users. | |||||
func (u *User) UploadAvatar(data []byte) error { | |||||
m, err := avatar.Prepare(data) | |||||
if err != nil { | |||||
return err | |||||
} | |||||
sess := x.NewSession() | |||||
defer sess.Close() | |||||
if err = sess.Begin(); err != nil { | |||||
return err | |||||
} | |||||
u.UseCustomAvatar = true | |||||
// Different users can upload same image as avatar | |||||
// If we prefix it with u.ID, it will be separated | |||||
// Otherwise, if any of the users delete his avatar | |||||
// Other users will lose their avatars too. | |||||
u.Avatar = fmt.Sprintf("%x", md5.Sum([]byte(fmt.Sprintf("%d-%x", u.ID, md5.Sum(data))))) | |||||
if err = updateUser(sess, u); err != nil { | |||||
return fmt.Errorf("updateUser: %v", err) | |||||
} | |||||
if err := storage.SaveFrom(storage.Avatars, u.CustomAvatarRelativePath(), func(w io.Writer) error { | |||||
if err := png.Encode(w, *m); err != nil { | |||||
log.Error("Encode: %v", err) | |||||
} | |||||
return err | |||||
}); err != nil { | |||||
return fmt.Errorf("Failed to create dir %s: %v", u.CustomAvatarRelativePath(), err) | |||||
} | |||||
return sess.Commit() | |||||
} | |||||
// DeleteAvatar deletes the user's custom avatar. | |||||
func (u *User) DeleteAvatar() error { | |||||
aPath := u.CustomAvatarRelativePath() | |||||
log.Trace("DeleteAvatar[%d]: %s", u.ID, aPath) | |||||
if len(u.Avatar) > 0 { | |||||
if err := storage.Avatars.Delete(aPath); err != nil { | |||||
return fmt.Errorf("Failed to remove %s: %v", aPath, err) | |||||
} | |||||
} | |||||
u.UseCustomAvatar = false | |||||
u.Avatar = "" | |||||
if _, err := x.ID(u.ID).Cols("avatar, use_custom_avatar").Update(u); err != nil { | |||||
return fmt.Errorf("UpdateUser: %v", err) | |||||
} | |||||
return nil | |||||
} |
@@ -9,6 +9,7 @@ import ( | |||||
"fmt" | "fmt" | ||||
"image" | "image" | ||||
"image/color/palette" | "image/color/palette" | ||||
// Enable PNG support: | // Enable PNG support: | ||||
_ "image/png" | _ "image/png" | ||||
"math/rand" | "math/rand" | ||||
@@ -57,11 +58,11 @@ func Prepare(data []byte) (*image.Image, error) { | |||||
if err != nil { | if err != nil { | ||||
return nil, fmt.Errorf("DecodeConfig: %v", err) | return nil, fmt.Errorf("DecodeConfig: %v", err) | ||||
} | } | ||||
if imgCfg.Width > setting.AvatarMaxWidth { | |||||
return nil, fmt.Errorf("Image width is too large: %d > %d", imgCfg.Width, setting.AvatarMaxWidth) | |||||
if imgCfg.Width > setting.Avatar.MaxWidth { | |||||
return nil, fmt.Errorf("Image width is too large: %d > %d", imgCfg.Width, setting.Avatar.MaxWidth) | |||||
} | } | ||||
if imgCfg.Height > setting.AvatarMaxHeight { | |||||
return nil, fmt.Errorf("Image height is too large: %d > %d", imgCfg.Height, setting.AvatarMaxHeight) | |||||
if imgCfg.Height > setting.Avatar.MaxHeight { | |||||
return nil, fmt.Errorf("Image height is too large: %d > %d", imgCfg.Height, setting.Avatar.MaxHeight) | |||||
} | } | ||||
img, _, err := image.Decode(bytes.NewReader(data)) | img, _, err := image.Decode(bytes.NewReader(data)) | ||||
@@ -22,8 +22,8 @@ func Test_RandomImage(t *testing.T) { | |||||
} | } | ||||
func Test_PrepareWithPNG(t *testing.T) { | func Test_PrepareWithPNG(t *testing.T) { | ||||
setting.AvatarMaxWidth = 4096 | |||||
setting.AvatarMaxHeight = 4096 | |||||
setting.Avatar.MaxWidth = 4096 | |||||
setting.Avatar.MaxHeight = 4096 | |||||
data, err := ioutil.ReadFile("testdata/avatar.png") | data, err := ioutil.ReadFile("testdata/avatar.png") | ||||
assert.NoError(t, err) | assert.NoError(t, err) | ||||
@@ -36,8 +36,8 @@ func Test_PrepareWithPNG(t *testing.T) { | |||||
} | } | ||||
func Test_PrepareWithJPEG(t *testing.T) { | func Test_PrepareWithJPEG(t *testing.T) { | ||||
setting.AvatarMaxWidth = 4096 | |||||
setting.AvatarMaxHeight = 4096 | |||||
setting.Avatar.MaxWidth = 4096 | |||||
setting.Avatar.MaxHeight = 4096 | |||||
data, err := ioutil.ReadFile("testdata/avatar.jpeg") | data, err := ioutil.ReadFile("testdata/avatar.jpeg") | ||||
assert.NoError(t, err) | assert.NoError(t, err) | ||||
@@ -50,15 +50,15 @@ func Test_PrepareWithJPEG(t *testing.T) { | |||||
} | } | ||||
func Test_PrepareWithInvalidImage(t *testing.T) { | func Test_PrepareWithInvalidImage(t *testing.T) { | ||||
setting.AvatarMaxWidth = 5 | |||||
setting.AvatarMaxHeight = 5 | |||||
setting.Avatar.MaxWidth = 5 | |||||
setting.Avatar.MaxHeight = 5 | |||||
_, err := Prepare([]byte{}) | _, err := Prepare([]byte{}) | ||||
assert.EqualError(t, err, "DecodeConfig: image: unknown format") | assert.EqualError(t, err, "DecodeConfig: image: unknown format") | ||||
} | } | ||||
func Test_PrepareWithInvalidImageSize(t *testing.T) { | func Test_PrepareWithInvalidImageSize(t *testing.T) { | ||||
setting.AvatarMaxWidth = 5 | |||||
setting.AvatarMaxHeight = 5 | |||||
setting.Avatar.MaxWidth = 5 | |||||
setting.Avatar.MaxHeight = 5 | |||||
data, err := ioutil.ReadFile("testdata/avatar.png") | data, err := ioutil.ReadFile("testdata/avatar.png") | ||||
assert.NoError(t, err) | assert.NoError(t, err) | ||||
@@ -47,7 +47,8 @@ var ( | |||||
ConnMaxLifetime time.Duration | ConnMaxLifetime time.Duration | ||||
IterateBufferSize int | IterateBufferSize int | ||||
}{ | }{ | ||||
Timeout: 500, | |||||
Timeout: 500, | |||||
IterateBufferSize: 50, | |||||
} | } | ||||
) | ) | ||||
@@ -0,0 +1,114 @@ | |||||
// Copyright 2020 The Gitea 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 setting | |||||
import ( | |||||
"net/url" | |||||
"code.gitea.io/gitea/modules/log" | |||||
"strk.kbt.io/projects/go/libravatar" | |||||
) | |||||
// settings | |||||
var ( | |||||
// Picture settings | |||||
Avatar = struct { | |||||
Storage | |||||
MaxWidth int | |||||
MaxHeight int | |||||
MaxFileSize int64 | |||||
}{ | |||||
MaxWidth: 4096, | |||||
MaxHeight: 3072, | |||||
MaxFileSize: 1048576, | |||||
} | |||||
GravatarSource string | |||||
GravatarSourceURL *url.URL | |||||
DisableGravatar bool | |||||
EnableFederatedAvatar bool | |||||
LibravatarService *libravatar.Libravatar | |||||
RepoAvatar = struct { | |||||
Storage | |||||
Fallback string | |||||
FallbackImage string | |||||
}{} | |||||
) | |||||
func newPictureService() { | |||||
sec := Cfg.Section("picture") | |||||
avatarSec := Cfg.Section("avatar") | |||||
storageType := sec.Key("AVATAR_STORAGE_TYPE").MustString("") | |||||
// Specifically default PATH to AVATAR_UPLOAD_PATH | |||||
avatarSec.Key("PATH").MustString( | |||||
sec.Key("AVATAR_UPLOAD_PATH").String()) | |||||
Avatar.Storage = getStorage("avatars", storageType, avatarSec) | |||||
Avatar.MaxWidth = sec.Key("AVATAR_MAX_WIDTH").MustInt(4096) | |||||
Avatar.MaxHeight = sec.Key("AVATAR_MAX_HEIGHT").MustInt(3072) | |||||
Avatar.MaxFileSize = sec.Key("AVATAR_MAX_FILE_SIZE").MustInt64(1048576) | |||||
switch source := sec.Key("GRAVATAR_SOURCE").MustString("gravatar"); source { | |||||
case "duoshuo": | |||||
GravatarSource = "http://gravatar.duoshuo.com/avatar/" | |||||
case "gravatar": | |||||
GravatarSource = "https://secure.gravatar.com/avatar/" | |||||
case "libravatar": | |||||
GravatarSource = "https://seccdn.libravatar.org/avatar/" | |||||
default: | |||||
GravatarSource = source | |||||
} | |||||
DisableGravatar = sec.Key("DISABLE_GRAVATAR").MustBool() | |||||
EnableFederatedAvatar = sec.Key("ENABLE_FEDERATED_AVATAR").MustBool(!InstallLock) | |||||
if OfflineMode { | |||||
DisableGravatar = true | |||||
EnableFederatedAvatar = false | |||||
} | |||||
if DisableGravatar { | |||||
EnableFederatedAvatar = false | |||||
} | |||||
if EnableFederatedAvatar || !DisableGravatar { | |||||
var err error | |||||
GravatarSourceURL, err = url.Parse(GravatarSource) | |||||
if err != nil { | |||||
log.Fatal("Failed to parse Gravatar URL(%s): %v", | |||||
GravatarSource, err) | |||||
} | |||||
} | |||||
if EnableFederatedAvatar { | |||||
LibravatarService = libravatar.New() | |||||
if GravatarSourceURL.Scheme == "https" { | |||||
LibravatarService.SetUseHTTPS(true) | |||||
LibravatarService.SetSecureFallbackHost(GravatarSourceURL.Host) | |||||
} else { | |||||
LibravatarService.SetUseHTTPS(false) | |||||
LibravatarService.SetFallbackHost(GravatarSourceURL.Host) | |||||
} | |||||
} | |||||
newRepoAvatarService() | |||||
} | |||||
func newRepoAvatarService() { | |||||
sec := Cfg.Section("picture") | |||||
repoAvatarSec := Cfg.Section("repo-avatar") | |||||
storageType := sec.Key("REPOSITORY_AVATAR_STORAGE_TYPE").MustString("") | |||||
// Specifically default PATH to AVATAR_UPLOAD_PATH | |||||
repoAvatarSec.Key("PATH").MustString( | |||||
sec.Key("REPOSITORY_AVATAR_UPLOAD_PATH").String()) | |||||
RepoAvatar.Storage = getStorage("repo-avatars", storageType, repoAvatarSec) | |||||
RepoAvatar.Fallback = sec.Key("REPOSITORY_AVATAR_FALLBACK").MustString("none") | |||||
RepoAvatar.FallbackImage = sec.Key("REPOSITORY_AVATAR_FALLBACK_IMAGE").MustString("/img/repo_default.png") | |||||
} |
@@ -30,7 +30,6 @@ import ( | |||||
"github.com/unknwon/com" | "github.com/unknwon/com" | ||||
gossh "golang.org/x/crypto/ssh" | gossh "golang.org/x/crypto/ssh" | ||||
ini "gopkg.in/ini.v1" | ini "gopkg.in/ini.v1" | ||||
"strk.kbt.io/projects/go/libravatar" | |||||
) | ) | ||||
// Scheme describes protocol types | // Scheme describes protocol types | ||||
@@ -272,20 +271,6 @@ var ( | |||||
DefaultEmailNotification string | DefaultEmailNotification string | ||||
} | } | ||||
// Picture settings | |||||
AvatarUploadPath string | |||||
AvatarMaxWidth int | |||||
AvatarMaxHeight int | |||||
GravatarSource string | |||||
GravatarSourceURL *url.URL | |||||
DisableGravatar bool | |||||
EnableFederatedAvatar bool | |||||
LibravatarService *libravatar.Libravatar | |||||
AvatarMaxFileSize int64 | |||||
RepositoryAvatarUploadPath string | |||||
RepositoryAvatarFallback string | |||||
RepositoryAvatarFallbackImage string | |||||
// Log settings | // Log settings | ||||
LogLevel string | LogLevel string | ||||
StacktraceLogLevel string | StacktraceLogLevel string | ||||
@@ -864,59 +849,7 @@ func NewContext() { | |||||
newRepository() | newRepository() | ||||
sec = Cfg.Section("picture") | |||||
AvatarUploadPath = sec.Key("AVATAR_UPLOAD_PATH").MustString(path.Join(AppDataPath, "avatars")) | |||||
forcePathSeparator(AvatarUploadPath) | |||||
if !filepath.IsAbs(AvatarUploadPath) { | |||||
AvatarUploadPath = path.Join(AppWorkPath, AvatarUploadPath) | |||||
} | |||||
RepositoryAvatarUploadPath = sec.Key("REPOSITORY_AVATAR_UPLOAD_PATH").MustString(path.Join(AppDataPath, "repo-avatars")) | |||||
forcePathSeparator(RepositoryAvatarUploadPath) | |||||
if !filepath.IsAbs(RepositoryAvatarUploadPath) { | |||||
RepositoryAvatarUploadPath = path.Join(AppWorkPath, RepositoryAvatarUploadPath) | |||||
} | |||||
RepositoryAvatarFallback = sec.Key("REPOSITORY_AVATAR_FALLBACK").MustString("none") | |||||
RepositoryAvatarFallbackImage = sec.Key("REPOSITORY_AVATAR_FALLBACK_IMAGE").MustString("/img/repo_default.png") | |||||
AvatarMaxWidth = sec.Key("AVATAR_MAX_WIDTH").MustInt(4096) | |||||
AvatarMaxHeight = sec.Key("AVATAR_MAX_HEIGHT").MustInt(3072) | |||||
AvatarMaxFileSize = sec.Key("AVATAR_MAX_FILE_SIZE").MustInt64(1048576) | |||||
switch source := sec.Key("GRAVATAR_SOURCE").MustString("gravatar"); source { | |||||
case "duoshuo": | |||||
GravatarSource = "http://gravatar.duoshuo.com/avatar/" | |||||
case "gravatar": | |||||
GravatarSource = "https://secure.gravatar.com/avatar/" | |||||
case "libravatar": | |||||
GravatarSource = "https://seccdn.libravatar.org/avatar/" | |||||
default: | |||||
GravatarSource = source | |||||
} | |||||
DisableGravatar = sec.Key("DISABLE_GRAVATAR").MustBool() | |||||
EnableFederatedAvatar = sec.Key("ENABLE_FEDERATED_AVATAR").MustBool(!InstallLock) | |||||
if OfflineMode { | |||||
DisableGravatar = true | |||||
EnableFederatedAvatar = false | |||||
} | |||||
if DisableGravatar { | |||||
EnableFederatedAvatar = false | |||||
} | |||||
if EnableFederatedAvatar || !DisableGravatar { | |||||
GravatarSourceURL, err = url.Parse(GravatarSource) | |||||
if err != nil { | |||||
log.Fatal("Failed to parse Gravatar URL(%s): %v", | |||||
GravatarSource, err) | |||||
} | |||||
} | |||||
if EnableFederatedAvatar { | |||||
LibravatarService = libravatar.New() | |||||
if GravatarSourceURL.Scheme == "https" { | |||||
LibravatarService.SetUseHTTPS(true) | |||||
LibravatarService.SetSecureFallbackHost(GravatarSourceURL.Host) | |||||
} else { | |||||
LibravatarService.SetUseHTTPS(false) | |||||
LibravatarService.SetFallbackHost(GravatarSourceURL.Host) | |||||
} | |||||
} | |||||
newPictureService() | |||||
if err = Cfg.Section("ui").MapTo(&UI); err != nil { | if err = Cfg.Section("ui").MapTo(&UI); err != nil { | ||||
log.Fatal("Failed to map UI settings: %v", err) | log.Fatal("Failed to map UI settings: %v", err) | ||||
@@ -82,12 +82,32 @@ func Copy(dstStorage ObjectStorage, dstPath string, srcStorage ObjectStorage, sr | |||||
return dstStorage.Save(dstPath, f) | return dstStorage.Save(dstPath, f) | ||||
} | } | ||||
// SaveFrom saves data to the ObjectStorage with path p from the callback | |||||
func SaveFrom(objStorage ObjectStorage, p string, callback func(w io.Writer) error) error { | |||||
pr, pw := io.Pipe() | |||||
defer pr.Close() | |||||
go func() { | |||||
defer pw.Close() | |||||
if err := callback(pw); err != nil { | |||||
_ = pw.CloseWithError(err) | |||||
} | |||||
}() | |||||
_, err := objStorage.Save(p, pr) | |||||
return err | |||||
} | |||||
var ( | var ( | ||||
// Attachments represents attachments storage | // Attachments represents attachments storage | ||||
Attachments ObjectStorage | Attachments ObjectStorage | ||||
// LFS represents lfs storage | // LFS represents lfs storage | ||||
LFS ObjectStorage | LFS ObjectStorage | ||||
// Avatars represents user avatars storage | |||||
Avatars ObjectStorage | |||||
// RepoAvatars represents repository avatars storage | |||||
RepoAvatars ObjectStorage | |||||
) | ) | ||||
// Init init the stoarge | // Init init the stoarge | ||||
@@ -96,6 +116,14 @@ func Init() error { | |||||
return err | return err | ||||
} | } | ||||
if err := initAvatars(); err != nil { | |||||
return err | |||||
} | |||||
if err := initRepoAvatars(); err != nil { | |||||
return err | |||||
} | |||||
return initLFS() | return initLFS() | ||||
} | } | ||||
@@ -112,6 +140,11 @@ func NewStorage(typStr string, cfg interface{}) (ObjectStorage, error) { | |||||
return fn(context.Background(), cfg) | return fn(context.Background(), cfg) | ||||
} | } | ||||
func initAvatars() (err error) { | |||||
Avatars, err = NewStorage(setting.Avatar.Storage.Type, setting.Avatar.Storage) | |||||
return | |||||
} | |||||
func initAttachments() (err error) { | func initAttachments() (err error) { | ||||
Attachments, err = NewStorage(setting.Attachment.Storage.Type, setting.Attachment.Storage) | Attachments, err = NewStorage(setting.Attachment.Storage.Type, setting.Attachment.Storage) | ||||
return | return | ||||
@@ -121,3 +154,8 @@ func initLFS() (err error) { | |||||
LFS, err = NewStorage(setting.LFS.Storage.Type, setting.LFS.Storage) | LFS, err = NewStorage(setting.LFS.Storage.Type, setting.LFS.Storage) | ||||
return | return | ||||
} | } | ||||
func initRepoAvatars() (err error) { | |||||
RepoAvatars, err = NewStorage(setting.RepoAvatar.Storage.Type, setting.RepoAvatar.Storage) | |||||
return | |||||
} |
@@ -30,7 +30,6 @@ import ( | |||||
mirror_service "code.gitea.io/gitea/services/mirror" | mirror_service "code.gitea.io/gitea/services/mirror" | ||||
repo_service "code.gitea.io/gitea/services/repository" | repo_service "code.gitea.io/gitea/services/repository" | ||||
"github.com/unknwon/com" | |||||
"mvdan.cc/xurls/v2" | "mvdan.cc/xurls/v2" | ||||
) | ) | ||||
@@ -928,7 +927,7 @@ func UpdateAvatarSetting(ctx *context.Context, form auth.AvatarForm) error { | |||||
// No avatar is uploaded and we not removing it here. | // No avatar is uploaded and we not removing it here. | ||||
// No random avatar generated here. | // No random avatar generated here. | ||||
// Just exit, no action. | // Just exit, no action. | ||||
if !com.IsFile(ctxRepo.CustomAvatarPath()) { | |||||
if ctxRepo.CustomAvatarRelativePath() == "" { | |||||
log.Trace("No avatar was uploaded for repo: %d. Default icon will appear instead.", ctxRepo.ID) | log.Trace("No avatar was uploaded for repo: %d. Default icon will appear instead.", ctxRepo.ID) | ||||
} | } | ||||
return nil | return nil | ||||
@@ -940,7 +939,7 @@ func UpdateAvatarSetting(ctx *context.Context, form auth.AvatarForm) error { | |||||
} | } | ||||
defer r.Close() | defer r.Close() | ||||
if form.Avatar.Size > setting.AvatarMaxFileSize { | |||||
if form.Avatar.Size > setting.Avatar.MaxFileSize { | |||||
return errors.New(ctx.Tr("settings.uploaded_avatar_is_too_big")) | return errors.New(ctx.Tr("settings.uploaded_avatar_is_too_big")) | ||||
} | } | ||||
@@ -7,8 +7,10 @@ package routes | |||||
import ( | import ( | ||||
"bytes" | "bytes" | ||||
"encoding/gob" | "encoding/gob" | ||||
"io" | |||||
"net/http" | "net/http" | ||||
"path" | "path" | ||||
"strings" | |||||
"text/template" | "text/template" | ||||
"time" | "time" | ||||
@@ -21,6 +23,7 @@ import ( | |||||
"code.gitea.io/gitea/modules/options" | "code.gitea.io/gitea/modules/options" | ||||
"code.gitea.io/gitea/modules/public" | "code.gitea.io/gitea/modules/public" | ||||
"code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
"code.gitea.io/gitea/modules/storage" | |||||
"code.gitea.io/gitea/modules/templates" | "code.gitea.io/gitea/modules/templates" | ||||
"code.gitea.io/gitea/modules/validation" | "code.gitea.io/gitea/modules/validation" | ||||
"code.gitea.io/gitea/routers" | "code.gitea.io/gitea/routers" | ||||
@@ -107,6 +110,61 @@ func RouterHandler(level log.Level) func(ctx *macaron.Context) { | |||||
} | } | ||||
} | } | ||||
func storageHandler(storageSetting setting.Storage, prefix string, objStore storage.ObjectStorage) macaron.Handler { | |||||
if storageSetting.ServeDirect { | |||||
return func(ctx *macaron.Context) { | |||||
req := ctx.Req.Request | |||||
if req.Method != "GET" && req.Method != "HEAD" { | |||||
return | |||||
} | |||||
if !strings.HasPrefix(req.RequestURI, "/"+prefix) { | |||||
return | |||||
} | |||||
rPath := strings.TrimPrefix(req.RequestURI, "/"+prefix) | |||||
u, err := objStore.URL(rPath, path.Base(rPath)) | |||||
if err != nil { | |||||
ctx.Error(500, err.Error()) | |||||
return | |||||
} | |||||
http.Redirect( | |||||
ctx.Resp, | |||||
req, | |||||
u.String(), | |||||
301, | |||||
) | |||||
} | |||||
} | |||||
return func(ctx *macaron.Context) { | |||||
req := ctx.Req.Request | |||||
if req.Method != "GET" && req.Method != "HEAD" { | |||||
return | |||||
} | |||||
if !strings.HasPrefix(req.RequestURI, "/"+prefix) { | |||||
return | |||||
} | |||||
rPath := strings.TrimPrefix(req.RequestURI, "/"+prefix) | |||||
rPath = strings.TrimPrefix(rPath, "/") | |||||
//If we have matched and access to release or issue | |||||
fr, err := objStore.Open(rPath) | |||||
if err != nil { | |||||
ctx.Error(500, err.Error()) | |||||
return | |||||
} | |||||
defer fr.Close() | |||||
_, err = io.Copy(ctx.Resp, fr) | |||||
if err != nil { | |||||
ctx.Error(500, err.Error()) | |||||
return | |||||
} | |||||
} | |||||
} | |||||
// NewMacaron initializes Macaron instance. | // NewMacaron initializes Macaron instance. | ||||
func NewMacaron() *macaron.Macaron { | func NewMacaron() *macaron.Macaron { | ||||
gob.Register(&u2f.Challenge{}) | gob.Register(&u2f.Challenge{}) | ||||
@@ -149,22 +207,9 @@ func NewMacaron() *macaron.Macaron { | |||||
ExpiresAfter: setting.StaticCacheTime, | ExpiresAfter: setting.StaticCacheTime, | ||||
}, | }, | ||||
)) | )) | ||||
m.Use(public.StaticHandler( | |||||
setting.AvatarUploadPath, | |||||
&public.Options{ | |||||
Prefix: "avatars", | |||||
SkipLogging: setting.DisableRouterLog, | |||||
ExpiresAfter: setting.StaticCacheTime, | |||||
}, | |||||
)) | |||||
m.Use(public.StaticHandler( | |||||
setting.RepositoryAvatarUploadPath, | |||||
&public.Options{ | |||||
Prefix: "repo-avatars", | |||||
SkipLogging: setting.DisableRouterLog, | |||||
ExpiresAfter: setting.StaticCacheTime, | |||||
}, | |||||
)) | |||||
m.Use(storageHandler(setting.Avatar.Storage, "avatars", storage.Avatars)) | |||||
m.Use(storageHandler(setting.RepoAvatar.Storage, "repo-avatars", storage.RepoAvatars)) | |||||
m.Use(templates.HTMLRenderer()) | m.Use(templates.HTMLRenderer()) | ||||
mailer.InitMailRender(templates.Mailer()) | mailer.InitMailRender(templates.Mailer()) | ||||
@@ -20,7 +20,6 @@ import ( | |||||
"code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
"code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
"github.com/unknwon/com" | |||||
"github.com/unknwon/i18n" | "github.com/unknwon/i18n" | ||||
) | ) | ||||
@@ -133,7 +132,7 @@ func UpdateAvatarSetting(ctx *context.Context, form auth.AvatarForm, ctxUser *mo | |||||
} | } | ||||
defer fr.Close() | defer fr.Close() | ||||
if form.Avatar.Size > setting.AvatarMaxFileSize { | |||||
if form.Avatar.Size > setting.Avatar.MaxFileSize { | |||||
return errors.New(ctx.Tr("settings.uploaded_avatar_is_too_big")) | return errors.New(ctx.Tr("settings.uploaded_avatar_is_too_big")) | ||||
} | } | ||||
@@ -147,7 +146,7 @@ func UpdateAvatarSetting(ctx *context.Context, form auth.AvatarForm, ctxUser *mo | |||||
if err = ctxUser.UploadAvatar(data); err != nil { | if err = ctxUser.UploadAvatar(data); err != nil { | ||||
return fmt.Errorf("UploadAvatar: %v", err) | return fmt.Errorf("UploadAvatar: %v", err) | ||||
} | } | ||||
} else if ctxUser.UseCustomAvatar && !com.IsFile(ctxUser.CustomAvatarPath()) { | |||||
} else if ctxUser.UseCustomAvatar && ctxUser.Avatar == "" { | |||||
// No avatar is uploaded but setting has been changed to enable, | // No avatar is uploaded but setting has been changed to enable, | ||||
// generate a random one when needed. | // generate a random one when needed. | ||||
if err := ctxUser.GenerateRandomAvatar(); err != nil { | if err := ctxUser.GenerateRandomAvatar(); err != nil { | ||||