2018-09-29 16:33:54 +08:00
|
|
|
// Copyright 2018 The Gitea Authors. All rights reserved.
|
2014-04-11 02:20:58 +08:00
|
|
|
// Copyright 2014 The Gogs Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2016-12-07 01:58:31 +08:00
|
|
|
package templates
|
2014-04-11 02:20:58 +08:00
|
|
|
|
|
|
|
import (
|
2017-03-02 08:25:44 +08:00
|
|
|
"bytes"
|
2014-04-11 02:20:58 +08:00
|
|
|
"container/list"
|
|
|
|
"encoding/json"
|
2017-12-04 07:14:26 +08:00
|
|
|
"errors"
|
2014-04-11 02:20:58 +08:00
|
|
|
"fmt"
|
2018-02-27 15:09:18 +08:00
|
|
|
"html"
|
2014-04-11 02:20:58 +08:00
|
|
|
"html/template"
|
2016-08-12 07:16:36 +08:00
|
|
|
"mime"
|
2017-11-28 17:43:51 +08:00
|
|
|
"net/url"
|
2016-08-12 07:16:36 +08:00
|
|
|
"path/filepath"
|
2019-11-07 21:34:28 +08:00
|
|
|
"regexp"
|
2014-05-22 09:37:13 +08:00
|
|
|
"runtime"
|
2014-04-11 02:20:58 +08:00
|
|
|
"strings"
|
2019-11-07 21:34:28 +08:00
|
|
|
texttmpl "text/template"
|
2014-04-11 02:20:58 +08:00
|
|
|
"time"
|
2019-11-01 12:48:30 +08:00
|
|
|
"unicode"
|
2014-05-26 08:11:25 +08:00
|
|
|
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/models"
|
|
|
|
"code.gitea.io/gitea/modules/base"
|
|
|
|
"code.gitea.io/gitea/modules/log"
|
2017-09-17 01:17:57 +08:00
|
|
|
"code.gitea.io/gitea/modules/markup"
|
2020-01-10 17:34:21 +08:00
|
|
|
"code.gitea.io/gitea/modules/repository"
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2019-08-15 22:46:21 +08:00
|
|
|
"code.gitea.io/gitea/modules/timeutil"
|
|
|
|
"code.gitea.io/gitea/modules/util"
|
2019-09-06 10:20:09 +08:00
|
|
|
"code.gitea.io/gitea/services/gitdiff"
|
2019-10-01 21:40:17 +08:00
|
|
|
mirror_service "code.gitea.io/gitea/services/mirror"
|
2017-11-22 15:09:48 +08:00
|
|
|
|
2019-10-16 05:24:16 +08:00
|
|
|
"github.com/editorconfig/editorconfig-core-go/v2"
|
2014-04-11 02:20:58 +08:00
|
|
|
)
|
|
|
|
|
2019-11-07 21:34:28 +08:00
|
|
|
// Used from static.go && dynamic.go
|
|
|
|
var mailSubjectSplit = regexp.MustCompile(`(?m)^-{3,}[\s]*$`)
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// NewFuncMap returns functions for injecting to templates
|
2016-03-07 05:40:04 +08:00
|
|
|
func NewFuncMap() []template.FuncMap {
|
|
|
|
return []template.FuncMap{map[string]interface{}{
|
|
|
|
"GoVer": func() string {
|
|
|
|
return strings.Title(runtime.Version())
|
|
|
|
},
|
|
|
|
"UseHTTPS": func() bool {
|
2016-11-27 18:14:25 +08:00
|
|
|
return strings.HasPrefix(setting.AppURL, "https")
|
2016-03-07 05:40:04 +08:00
|
|
|
},
|
|
|
|
"AppName": func() string {
|
|
|
|
return setting.AppName
|
|
|
|
},
|
|
|
|
"AppSubUrl": func() string {
|
2016-11-27 18:14:25 +08:00
|
|
|
return setting.AppSubURL
|
2016-03-07 05:40:04 +08:00
|
|
|
},
|
2019-10-22 20:11:01 +08:00
|
|
|
"StaticUrlPrefix": func() string {
|
|
|
|
return setting.StaticURLPrefix
|
|
|
|
},
|
2016-03-07 05:40:04 +08:00
|
|
|
"AppUrl": func() string {
|
2016-11-27 18:14:25 +08:00
|
|
|
return setting.AppURL
|
2016-03-07 05:40:04 +08:00
|
|
|
},
|
|
|
|
"AppVer": func() string {
|
|
|
|
return setting.AppVer
|
|
|
|
},
|
2017-02-28 08:40:02 +08:00
|
|
|
"AppBuiltWith": func() string {
|
2017-03-01 09:45:21 +08:00
|
|
|
return setting.AppBuiltWith
|
2017-02-28 08:40:02 +08:00
|
|
|
},
|
2016-03-07 05:40:04 +08:00
|
|
|
"AppDomain": func() string {
|
|
|
|
return setting.Domain
|
|
|
|
},
|
|
|
|
"DisableGravatar": func() bool {
|
|
|
|
return setting.DisableGravatar
|
|
|
|
},
|
2019-05-08 16:41:35 +08:00
|
|
|
"DefaultShowFullName": func() bool {
|
|
|
|
return setting.UI.DefaultShowFullName
|
|
|
|
},
|
2016-09-01 13:01:32 +08:00
|
|
|
"ShowFooterTemplateLoadTime": func() bool {
|
|
|
|
return setting.ShowFooterTemplateLoadTime
|
|
|
|
},
|
2016-03-07 05:40:04 +08:00
|
|
|
"LoadTimes": func(startTime time.Time) string {
|
|
|
|
return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms"
|
|
|
|
},
|
2019-12-28 07:43:56 +08:00
|
|
|
"AllowedReactions": func() []string {
|
|
|
|
return setting.UI.Reactions
|
|
|
|
},
|
2017-12-11 12:37:04 +08:00
|
|
|
"AvatarLink": base.AvatarLink,
|
|
|
|
"Safe": Safe,
|
|
|
|
"SafeJS": SafeJS,
|
|
|
|
"Str2html": Str2html,
|
2019-08-15 22:46:21 +08:00
|
|
|
"TimeSince": timeutil.TimeSince,
|
|
|
|
"TimeSinceUnix": timeutil.TimeSinceUnix,
|
|
|
|
"RawTimeSince": timeutil.RawTimeSince,
|
2017-12-11 12:37:04 +08:00
|
|
|
"FileSize": base.FileSize,
|
2020-02-04 03:50:37 +08:00
|
|
|
"PrettyNumber": base.PrettyNumber,
|
2017-12-11 12:37:04 +08:00
|
|
|
"Subtract": base.Subtract,
|
2018-05-01 15:04:36 +08:00
|
|
|
"EntryIcon": base.EntryIcon,
|
2019-07-08 10:14:12 +08:00
|
|
|
"MigrationIcon": MigrationIcon,
|
2016-03-07 05:40:04 +08:00
|
|
|
"Add": func(a, b int) int {
|
|
|
|
return a + b
|
|
|
|
},
|
|
|
|
"ActionIcon": ActionIcon,
|
|
|
|
"DateFmtLong": func(t time.Time) string {
|
|
|
|
return t.Format(time.RFC1123Z)
|
|
|
|
},
|
|
|
|
"DateFmtShort": func(t time.Time) string {
|
|
|
|
return t.Format("Jan 02, 2006")
|
|
|
|
},
|
2019-07-22 16:58:26 +08:00
|
|
|
"SizeFmt": base.FileSize,
|
|
|
|
"List": List,
|
2016-03-07 05:40:04 +08:00
|
|
|
"SubStr": func(str string, start, length int) string {
|
|
|
|
if len(str) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
end := start + length
|
|
|
|
if length == -1 {
|
|
|
|
end = len(str)
|
|
|
|
}
|
|
|
|
if len(str) < end {
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
return str[start:end]
|
|
|
|
},
|
2018-08-29 21:43:58 +08:00
|
|
|
"EllipsisString": base.EllipsisString,
|
|
|
|
"DiffTypeToStr": DiffTypeToStr,
|
|
|
|
"DiffLineTypeToStr": DiffLineTypeToStr,
|
|
|
|
"Sha1": Sha1,
|
|
|
|
"ShortSha": base.ShortSha,
|
|
|
|
"MD5": base.EncodeMD5,
|
2016-03-07 05:40:04 +08:00
|
|
|
"ActionContent2Commits": ActionContent2Commits,
|
2017-11-28 17:43:51 +08:00
|
|
|
"PathEscape": url.PathEscape,
|
2016-03-07 05:40:04 +08:00
|
|
|
"EscapePound": func(str string) string {
|
2016-09-18 23:46:52 +08:00
|
|
|
return strings.NewReplacer("%", "%25", "#", "%23", " ", "%20", "?", "%3F").Replace(str)
|
2016-03-07 05:40:04 +08:00
|
|
|
},
|
2019-09-10 17:03:30 +08:00
|
|
|
"PathEscapeSegments": util.PathEscapeSegments,
|
|
|
|
"URLJoin": util.URLJoin,
|
|
|
|
"RenderCommitMessage": RenderCommitMessage,
|
|
|
|
"RenderCommitMessageLink": RenderCommitMessageLink,
|
|
|
|
"RenderCommitMessageLinkSubject": RenderCommitMessageLinkSubject,
|
|
|
|
"RenderCommitBody": RenderCommitBody,
|
|
|
|
"RenderNote": RenderNote,
|
|
|
|
"IsMultilineCommitMessage": IsMultilineCommitMessage,
|
2016-03-07 05:40:04 +08:00
|
|
|
"ThemeColorMetaTag": func() string {
|
2016-07-24 00:23:54 +08:00
|
|
|
return setting.UI.ThemeColorMetaTag
|
2016-03-07 05:40:04 +08:00
|
|
|
},
|
2017-04-01 09:03:01 +08:00
|
|
|
"MetaAuthor": func() string {
|
|
|
|
return setting.UI.Meta.Author
|
|
|
|
},
|
|
|
|
"MetaDescription": func() string {
|
|
|
|
return setting.UI.Meta.Description
|
|
|
|
},
|
|
|
|
"MetaKeywords": func() string {
|
|
|
|
return setting.UI.Meta.Keywords
|
|
|
|
},
|
2019-11-22 04:06:23 +08:00
|
|
|
"UseServiceWorker": func() bool {
|
|
|
|
return setting.UI.UseServiceWorker
|
|
|
|
},
|
2016-08-12 07:16:36 +08:00
|
|
|
"FilenameIsImage": func(filename string) bool {
|
|
|
|
mimeType := mime.TypeByExtension(filepath.Ext(filename))
|
|
|
|
return strings.HasPrefix(mimeType, "image/")
|
|
|
|
},
|
2016-08-12 08:07:09 +08:00
|
|
|
"TabSizeClass": func(ec *editorconfig.Editorconfig, filename string) string {
|
|
|
|
if ec != nil {
|
2019-10-16 05:24:16 +08:00
|
|
|
def, err := ec.GetDefinitionForFilename(filename)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("tab size class: getting definition for filename: %v", err)
|
|
|
|
return "tab-size-8"
|
|
|
|
}
|
2016-08-12 08:07:09 +08:00
|
|
|
if def.TabWidth > 0 {
|
|
|
|
return fmt.Sprintf("tab-size-%d", def.TabWidth)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "tab-size-8"
|
|
|
|
},
|
2016-12-29 00:35:52 +08:00
|
|
|
"SubJumpablePath": func(str string) []string {
|
|
|
|
var path []string
|
|
|
|
index := strings.LastIndex(str, "/")
|
|
|
|
if index != -1 && index != len(str) {
|
2019-05-28 23:45:54 +08:00
|
|
|
path = append(path, str[0:index+1], str[index+1:])
|
2016-12-29 00:35:52 +08:00
|
|
|
} else {
|
|
|
|
path = append(path, str)
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
},
|
2020-01-20 18:07:30 +08:00
|
|
|
"Json": func(in interface{}) string {
|
|
|
|
out, err := json.Marshal(in)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return string(out)
|
|
|
|
},
|
2017-03-02 08:25:44 +08:00
|
|
|
"JsonPrettyPrint": func(in string) string {
|
|
|
|
var out bytes.Buffer
|
|
|
|
err := json.Indent(&out, []byte(in), "", " ")
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return out.String()
|
|
|
|
},
|
2017-09-12 17:25:42 +08:00
|
|
|
"DisableGitHooks": func() bool {
|
|
|
|
return setting.DisableGitHooks
|
|
|
|
},
|
2018-08-24 13:00:22 +08:00
|
|
|
"DisableImportLocal": func() bool {
|
|
|
|
return !setting.ImportLocalPaths
|
|
|
|
},
|
2017-10-15 07:17:39 +08:00
|
|
|
"TrN": TrN,
|
2017-12-04 07:14:26 +08:00
|
|
|
"Dict": func(values ...interface{}) (map[string]interface{}, error) {
|
|
|
|
if len(values)%2 != 0 {
|
|
|
|
return nil, errors.New("invalid dict call")
|
|
|
|
}
|
|
|
|
dict := make(map[string]interface{}, len(values)/2)
|
|
|
|
for i := 0; i < len(values); i += 2 {
|
|
|
|
key, ok := values[i].(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("dict keys must be strings")
|
|
|
|
}
|
|
|
|
dict[key] = values[i+1]
|
|
|
|
}
|
|
|
|
return dict, nil
|
|
|
|
},
|
2018-04-29 13:58:47 +08:00
|
|
|
"Printf": fmt.Sprintf,
|
|
|
|
"Escape": Escape,
|
|
|
|
"Sec2Time": models.SecToTime,
|
2018-05-02 03:05:28 +08:00
|
|
|
"ParseDeadline": func(deadline string) []string {
|
|
|
|
return strings.Split(deadline, "|")
|
|
|
|
},
|
2018-07-06 05:25:04 +08:00
|
|
|
"DefaultTheme": func() string {
|
|
|
|
return setting.UI.DefaultTheme
|
|
|
|
},
|
2018-08-06 12:43:22 +08:00
|
|
|
"dict": func(values ...interface{}) (map[string]interface{}, error) {
|
|
|
|
if len(values) == 0 {
|
|
|
|
return nil, errors.New("invalid dict call")
|
|
|
|
}
|
|
|
|
|
|
|
|
dict := make(map[string]interface{})
|
|
|
|
|
|
|
|
for i := 0; i < len(values); i++ {
|
|
|
|
switch key := values[i].(type) {
|
|
|
|
case string:
|
|
|
|
i++
|
|
|
|
if i == len(values) {
|
|
|
|
return nil, errors.New("specify the key for non array values")
|
|
|
|
}
|
|
|
|
dict[key] = values[i]
|
|
|
|
case map[string]interface{}:
|
|
|
|
m := values[i].(map[string]interface{})
|
|
|
|
for i, v := range m {
|
|
|
|
dict[i] = v
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return nil, errors.New("dict values must be maps")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dict, nil
|
|
|
|
},
|
2019-05-06 00:25:25 +08:00
|
|
|
"percentage": func(n int, values ...int) float32 {
|
|
|
|
var sum = 0
|
|
|
|
for i := 0; i < len(values); i++ {
|
|
|
|
sum += values[i]
|
|
|
|
}
|
|
|
|
return float32(n) * 100 / float32(sum)
|
|
|
|
},
|
2019-09-06 10:20:09 +08:00
|
|
|
"CommentMustAsDiff": gitdiff.CommentMustAsDiff,
|
2019-10-01 21:40:17 +08:00
|
|
|
"MirrorAddress": mirror_service.Address,
|
|
|
|
"MirrorFullAddress": mirror_service.AddressNoCredentials,
|
2019-10-09 21:09:02 +08:00
|
|
|
"MirrorUserName": mirror_service.Username,
|
|
|
|
"MirrorPassword": mirror_service.Password,
|
2019-11-02 06:02:41 +08:00
|
|
|
"CommitType": func(commit interface{}) string {
|
|
|
|
switch commit.(type) {
|
|
|
|
case models.SignCommitWithStatuses:
|
|
|
|
return "SignCommitWithStatuses"
|
|
|
|
case models.SignCommit:
|
|
|
|
return "SignCommit"
|
|
|
|
case models.UserCommit:
|
|
|
|
return "UserCommit"
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
},
|
2019-12-28 22:43:46 +08:00
|
|
|
"contain": func(s []int64, id int64) bool {
|
|
|
|
for i := 0; i < len(s); i++ {
|
|
|
|
if s[i] == id {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
},
|
2020-02-12 01:02:41 +08:00
|
|
|
"svg": func(icon string, size int) template.HTML {
|
2020-02-18 07:11:59 +08:00
|
|
|
return template.HTML(fmt.Sprintf(`<svg class="svg %s" width="%d" height="%d" aria-hidden="true"><use xlink:href="#%s" /></svg>`, icon, size, size, icon))
|
2020-02-12 01:02:41 +08:00
|
|
|
},
|
2016-03-07 05:40:04 +08:00
|
|
|
}}
|
2015-11-14 11:45:33 +08:00
|
|
|
}
|
|
|
|
|
2019-11-07 21:34:28 +08:00
|
|
|
// NewTextFuncMap returns functions for injecting to text templates
|
|
|
|
// It's a subset of those used for HTML and other templates
|
|
|
|
func NewTextFuncMap() []texttmpl.FuncMap {
|
|
|
|
return []texttmpl.FuncMap{map[string]interface{}{
|
|
|
|
"GoVer": func() string {
|
|
|
|
return strings.Title(runtime.Version())
|
|
|
|
},
|
|
|
|
"AppName": func() string {
|
|
|
|
return setting.AppName
|
|
|
|
},
|
|
|
|
"AppSubUrl": func() string {
|
|
|
|
return setting.AppSubURL
|
|
|
|
},
|
|
|
|
"AppUrl": func() string {
|
|
|
|
return setting.AppURL
|
|
|
|
},
|
|
|
|
"AppVer": func() string {
|
|
|
|
return setting.AppVer
|
|
|
|
},
|
|
|
|
"AppBuiltWith": func() string {
|
|
|
|
return setting.AppBuiltWith
|
|
|
|
},
|
|
|
|
"AppDomain": func() string {
|
|
|
|
return setting.Domain
|
|
|
|
},
|
|
|
|
"TimeSince": timeutil.TimeSince,
|
|
|
|
"TimeSinceUnix": timeutil.TimeSinceUnix,
|
|
|
|
"RawTimeSince": timeutil.RawTimeSince,
|
|
|
|
"DateFmtLong": func(t time.Time) string {
|
|
|
|
return t.Format(time.RFC1123Z)
|
|
|
|
},
|
|
|
|
"DateFmtShort": func(t time.Time) string {
|
|
|
|
return t.Format("Jan 02, 2006")
|
|
|
|
},
|
|
|
|
"List": List,
|
|
|
|
"SubStr": func(str string, start, length int) string {
|
|
|
|
if len(str) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
end := start + length
|
|
|
|
if length == -1 {
|
|
|
|
end = len(str)
|
|
|
|
}
|
|
|
|
if len(str) < end {
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
return str[start:end]
|
|
|
|
},
|
|
|
|
"EllipsisString": base.EllipsisString,
|
|
|
|
"URLJoin": util.URLJoin,
|
|
|
|
"Dict": func(values ...interface{}) (map[string]interface{}, error) {
|
|
|
|
if len(values)%2 != 0 {
|
|
|
|
return nil, errors.New("invalid dict call")
|
|
|
|
}
|
|
|
|
dict := make(map[string]interface{}, len(values)/2)
|
|
|
|
for i := 0; i < len(values); i += 2 {
|
|
|
|
key, ok := values[i].(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("dict keys must be strings")
|
|
|
|
}
|
|
|
|
dict[key] = values[i+1]
|
|
|
|
}
|
|
|
|
return dict, nil
|
|
|
|
},
|
|
|
|
"Printf": fmt.Sprintf,
|
|
|
|
"Escape": Escape,
|
|
|
|
"Sec2Time": models.SecToTime,
|
|
|
|
"ParseDeadline": func(deadline string) []string {
|
|
|
|
return strings.Split(deadline, "|")
|
|
|
|
},
|
|
|
|
"dict": func(values ...interface{}) (map[string]interface{}, error) {
|
|
|
|
if len(values) == 0 {
|
|
|
|
return nil, errors.New("invalid dict call")
|
|
|
|
}
|
|
|
|
|
|
|
|
dict := make(map[string]interface{})
|
|
|
|
|
|
|
|
for i := 0; i < len(values); i++ {
|
|
|
|
switch key := values[i].(type) {
|
|
|
|
case string:
|
|
|
|
i++
|
|
|
|
if i == len(values) {
|
|
|
|
return nil, errors.New("specify the key for non array values")
|
|
|
|
}
|
|
|
|
dict[key] = values[i]
|
|
|
|
case map[string]interface{}:
|
|
|
|
m := values[i].(map[string]interface{})
|
|
|
|
for i, v := range m {
|
|
|
|
dict[i] = v
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return nil, errors.New("dict values must be maps")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dict, nil
|
|
|
|
},
|
|
|
|
"percentage": func(n int, values ...int) float32 {
|
|
|
|
var sum = 0
|
|
|
|
for i := 0; i < len(values); i++ {
|
|
|
|
sum += values[i]
|
|
|
|
}
|
|
|
|
return float32(n) * 100 / float32(sum)
|
|
|
|
},
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// Safe render raw as HTML
|
2015-08-08 17:10:34 +08:00
|
|
|
func Safe(raw string) template.HTML {
|
|
|
|
return template.HTML(raw)
|
|
|
|
}
|
|
|
|
|
2017-08-23 22:58:05 +08:00
|
|
|
// SafeJS renders raw as JS
|
|
|
|
func SafeJS(raw string) template.JS {
|
|
|
|
return template.JS(raw)
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// Str2html render Markdown text to HTML
|
2014-04-11 02:20:58 +08:00
|
|
|
func Str2html(raw string) template.HTML {
|
2017-09-17 01:17:57 +08:00
|
|
|
return template.HTML(markup.Sanitize(raw))
|
2014-04-11 02:20:58 +08:00
|
|
|
}
|
|
|
|
|
2018-02-11 21:42:28 +08:00
|
|
|
// Escape escapes a HTML string
|
|
|
|
func Escape(raw string) string {
|
|
|
|
return html.EscapeString(raw)
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// List traversings the list
|
2014-04-11 02:20:58 +08:00
|
|
|
func List(l *list.List) chan interface{} {
|
|
|
|
e := l.Front()
|
|
|
|
c := make(chan interface{})
|
|
|
|
go func() {
|
|
|
|
for e != nil {
|
|
|
|
c <- e.Value
|
|
|
|
e = e.Next()
|
|
|
|
}
|
|
|
|
close(c)
|
|
|
|
}()
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// Sha1 returns sha1 sum of string
|
2015-02-19 05:52:22 +08:00
|
|
|
func Sha1(str string) string {
|
2015-11-14 06:10:25 +08:00
|
|
|
return base.EncodeSha1(str)
|
2014-12-09 15:18:25 +08:00
|
|
|
}
|
|
|
|
|
2015-01-31 07:05:20 +08:00
|
|
|
// RenderCommitMessage renders commit message with XSS-safe and special links.
|
2017-11-13 09:35:55 +08:00
|
|
|
func RenderCommitMessage(msg, urlPrefix string, metas map[string]string) template.HTML {
|
2018-02-27 15:09:18 +08:00
|
|
|
return RenderCommitMessageLink(msg, urlPrefix, "", metas)
|
2017-11-13 09:35:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// RenderCommitMessageLink renders commit message as a XXS-safe link to the provided
|
|
|
|
// default url, handling for special links.
|
2018-02-27 15:09:18 +08:00
|
|
|
func RenderCommitMessageLink(msg, urlPrefix, urlDefault string, metas map[string]string) template.HTML {
|
2015-09-19 09:57:06 +08:00
|
|
|
cleanMsg := template.HTMLEscapeString(msg)
|
2018-02-27 15:09:18 +08:00
|
|
|
// we can safely assume that it will not return any error, since there
|
|
|
|
// shouldn't be any special HTML.
|
|
|
|
fullMessage, err := markup.RenderCommitMessage([]byte(cleanMsg), urlPrefix, urlDefault, metas)
|
|
|
|
if err != nil {
|
2019-04-02 15:48:31 +08:00
|
|
|
log.Error("RenderCommitMessage: %v", err)
|
2018-02-27 15:09:18 +08:00
|
|
|
return ""
|
|
|
|
}
|
|
|
|
msgLines := strings.Split(strings.TrimSpace(string(fullMessage)), "\n")
|
2017-11-13 09:35:55 +08:00
|
|
|
if len(msgLines) == 0 {
|
2015-12-07 07:18:12 +08:00
|
|
|
return template.HTML("")
|
2015-09-19 09:57:06 +08:00
|
|
|
}
|
2017-11-13 09:35:55 +08:00
|
|
|
return template.HTML(msgLines[0])
|
2015-01-31 07:05:20 +08:00
|
|
|
}
|
|
|
|
|
2019-09-10 17:03:30 +08:00
|
|
|
// RenderCommitMessageLinkSubject renders commit message as a XXS-safe link to
|
|
|
|
// the provided default url, handling for special links without email to links.
|
|
|
|
func RenderCommitMessageLinkSubject(msg, urlPrefix, urlDefault string, metas map[string]string) template.HTML {
|
2019-11-01 12:48:30 +08:00
|
|
|
msgLine := strings.TrimLeftFunc(msg, unicode.IsSpace)
|
|
|
|
lineEnd := strings.IndexByte(msgLine, '\n')
|
|
|
|
if lineEnd > 0 {
|
|
|
|
msgLine = msgLine[:lineEnd]
|
|
|
|
}
|
|
|
|
msgLine = strings.TrimRightFunc(msgLine, unicode.IsSpace)
|
|
|
|
if len(msgLine) == 0 {
|
|
|
|
return template.HTML("")
|
|
|
|
}
|
|
|
|
|
2019-09-10 17:03:30 +08:00
|
|
|
// we can safely assume that it will not return any error, since there
|
|
|
|
// shouldn't be any special HTML.
|
2019-11-01 12:48:30 +08:00
|
|
|
renderedMessage, err := markup.RenderCommitMessageSubject([]byte(template.HTMLEscapeString(msgLine)), urlPrefix, urlDefault, metas)
|
2019-09-10 17:03:30 +08:00
|
|
|
if err != nil {
|
|
|
|
log.Error("RenderCommitMessageSubject: %v", err)
|
|
|
|
return template.HTML("")
|
|
|
|
}
|
2019-11-01 12:48:30 +08:00
|
|
|
return template.HTML(renderedMessage)
|
2019-09-10 17:03:30 +08:00
|
|
|
}
|
|
|
|
|
2017-11-30 13:08:40 +08:00
|
|
|
// RenderCommitBody extracts the body of a commit message without its title.
|
|
|
|
func RenderCommitBody(msg, urlPrefix string, metas map[string]string) template.HTML {
|
2019-11-01 12:48:30 +08:00
|
|
|
msgLine := strings.TrimRightFunc(msg, unicode.IsSpace)
|
|
|
|
lineEnd := strings.IndexByte(msgLine, '\n')
|
|
|
|
if lineEnd > 0 {
|
|
|
|
msgLine = msgLine[lineEnd+1:]
|
|
|
|
} else {
|
|
|
|
return template.HTML("")
|
|
|
|
}
|
|
|
|
msgLine = strings.TrimLeftFunc(msgLine, unicode.IsSpace)
|
|
|
|
if len(msgLine) == 0 {
|
|
|
|
return template.HTML("")
|
|
|
|
}
|
|
|
|
|
|
|
|
renderedMessage, err := markup.RenderCommitMessage([]byte(template.HTMLEscapeString(msgLine)), urlPrefix, "", metas)
|
2018-02-27 15:09:18 +08:00
|
|
|
if err != nil {
|
2019-04-02 15:48:31 +08:00
|
|
|
log.Error("RenderCommitMessage: %v", err)
|
2018-02-27 15:09:18 +08:00
|
|
|
return ""
|
|
|
|
}
|
2019-11-01 12:48:30 +08:00
|
|
|
return template.HTML(renderedMessage)
|
2017-11-30 13:08:40 +08:00
|
|
|
}
|
|
|
|
|
2019-05-24 15:52:05 +08:00
|
|
|
// RenderNote renders the contents of a git-notes file as a commit message.
|
|
|
|
func RenderNote(msg, urlPrefix string, metas map[string]string) template.HTML {
|
|
|
|
cleanMsg := template.HTMLEscapeString(msg)
|
|
|
|
fullMessage, err := markup.RenderCommitMessage([]byte(cleanMsg), urlPrefix, "", metas)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("RenderNote: %v", err)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return template.HTML(string(fullMessage))
|
|
|
|
}
|
|
|
|
|
2017-11-30 13:08:40 +08:00
|
|
|
// IsMultilineCommitMessage checks to see if a commit message contains multiple lines.
|
|
|
|
func IsMultilineCommitMessage(msg string) bool {
|
2018-06-15 22:07:48 +08:00
|
|
|
return strings.Count(strings.TrimSpace(msg), "\n") >= 1
|
2017-11-30 13:08:40 +08:00
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// Actioner describes an action
|
2014-04-11 02:20:58 +08:00
|
|
|
type Actioner interface {
|
2017-09-20 09:22:42 +08:00
|
|
|
GetOpType() models.ActionType
|
2014-04-11 02:20:58 +08:00
|
|
|
GetActUserName() string
|
2014-05-09 14:42:50 +08:00
|
|
|
GetRepoUserName() string
|
2014-04-11 02:20:58 +08:00
|
|
|
GetRepoName() string
|
2015-09-01 21:29:52 +08:00
|
|
|
GetRepoPath() string
|
|
|
|
GetRepoLink() string
|
2014-04-11 02:20:58 +08:00
|
|
|
GetBranch() string
|
|
|
|
GetContent() string
|
2015-09-01 21:29:52 +08:00
|
|
|
GetCreate() time.Time
|
|
|
|
GetIssueInfos() []string
|
2014-04-11 02:20:58 +08:00
|
|
|
}
|
|
|
|
|
2017-09-20 09:22:42 +08:00
|
|
|
// ActionIcon accepts an action operation type and returns an icon class name.
|
|
|
|
func ActionIcon(opType models.ActionType) string {
|
2014-04-11 02:20:58 +08:00
|
|
|
switch opType {
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionCreateRepo, models.ActionTransferRepo:
|
2014-07-26 12:24:27 +08:00
|
|
|
return "repo"
|
2017-09-21 15:43:26 +08:00
|
|
|
case models.ActionCommitRepo, models.ActionPushTag, models.ActionDeleteTag, models.ActionDeleteBranch:
|
2014-07-26 12:24:27 +08:00
|
|
|
return "git-commit"
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionCreateIssue:
|
2014-07-26 12:24:27 +08:00
|
|
|
return "issue-opened"
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionCreatePullRequest:
|
2015-11-17 00:39:48 +08:00
|
|
|
return "git-pull-request"
|
2019-12-22 16:29:26 +08:00
|
|
|
case models.ActionCommentIssue, models.ActionCommentPull:
|
2016-07-16 12:45:13 +08:00
|
|
|
return "comment-discussion"
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionMergePullRequest:
|
2015-11-17 00:39:48 +08:00
|
|
|
return "git-merge"
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionCloseIssue, models.ActionClosePullRequest:
|
2016-02-23 01:40:00 +08:00
|
|
|
return "issue-closed"
|
2017-09-20 09:22:42 +08:00
|
|
|
case models.ActionReopenIssue, models.ActionReopenPullRequest:
|
2016-03-06 01:58:51 +08:00
|
|
|
return "issue-reopened"
|
2018-09-07 10:06:09 +08:00
|
|
|
case models.ActionMirrorSyncPush, models.ActionMirrorSyncCreate, models.ActionMirrorSyncDelete:
|
|
|
|
return "repo-clone"
|
2019-11-15 07:52:18 +08:00
|
|
|
case models.ActionApprovePullRequest:
|
|
|
|
return "eye"
|
|
|
|
case models.ActionRejectPullRequest:
|
|
|
|
return "x"
|
2014-04-11 02:20:58 +08:00
|
|
|
default:
|
|
|
|
return "invalid type"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// ActionContent2Commits converts action content to push commits
|
2020-01-10 17:34:21 +08:00
|
|
|
func ActionContent2Commits(act Actioner) *repository.PushCommits {
|
|
|
|
push := repository.NewPushCommits()
|
2015-11-14 06:10:25 +08:00
|
|
|
if err := json.Unmarshal([]byte(act.GetContent()), push); err != nil {
|
2019-04-02 15:48:31 +08:00
|
|
|
log.Error("json.Unmarshal:\n%s\nERROR: %v", act.GetContent(), err)
|
2014-07-26 12:24:27 +08:00
|
|
|
}
|
|
|
|
return push
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// DiffTypeToStr returns diff type name
|
2014-04-11 02:20:58 +08:00
|
|
|
func DiffTypeToStr(diffType int) string {
|
|
|
|
diffTypes := map[int]string{
|
2015-11-03 22:52:17 +08:00
|
|
|
1: "add", 2: "modify", 3: "del", 4: "rename",
|
2014-04-11 02:20:58 +08:00
|
|
|
}
|
|
|
|
return diffTypes[diffType]
|
|
|
|
}
|
|
|
|
|
2016-11-25 14:23:48 +08:00
|
|
|
// DiffLineTypeToStr returns diff line type name
|
2014-04-11 02:20:58 +08:00
|
|
|
func DiffLineTypeToStr(diffType int) string {
|
|
|
|
switch diffType {
|
|
|
|
case 2:
|
|
|
|
return "add"
|
|
|
|
case 3:
|
|
|
|
return "del"
|
|
|
|
case 4:
|
|
|
|
return "tag"
|
|
|
|
}
|
|
|
|
return "same"
|
|
|
|
}
|
2017-10-15 07:17:39 +08:00
|
|
|
|
|
|
|
// Language specific rules for translating plural texts
|
|
|
|
var trNLangRules = map[string]func(int64) int{
|
|
|
|
"en-US": func(cnt int64) int {
|
|
|
|
if cnt == 1 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return 1
|
|
|
|
},
|
|
|
|
"lv-LV": func(cnt int64) int {
|
|
|
|
if cnt%10 == 1 && cnt%100 != 11 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return 1
|
|
|
|
},
|
|
|
|
"ru-RU": func(cnt int64) int {
|
|
|
|
if cnt%10 == 1 && cnt%100 != 11 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return 1
|
|
|
|
},
|
|
|
|
"zh-CN": func(cnt int64) int {
|
|
|
|
return 0
|
|
|
|
},
|
|
|
|
"zh-HK": func(cnt int64) int {
|
|
|
|
return 0
|
|
|
|
},
|
|
|
|
"zh-TW": func(cnt int64) int {
|
|
|
|
return 0
|
|
|
|
},
|
2019-05-05 06:44:43 +08:00
|
|
|
"fr-FR": func(cnt int64) int {
|
|
|
|
if cnt > -2 && cnt < 2 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return 1
|
|
|
|
},
|
2017-10-15 07:17:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// TrN returns key to be used for plural text translation
|
|
|
|
func TrN(lang string, cnt interface{}, key1, keyN string) string {
|
|
|
|
var c int64
|
|
|
|
if t, ok := cnt.(int); ok {
|
|
|
|
c = int64(t)
|
|
|
|
} else if t, ok := cnt.(int16); ok {
|
|
|
|
c = int64(t)
|
|
|
|
} else if t, ok := cnt.(int32); ok {
|
|
|
|
c = int64(t)
|
|
|
|
} else if t, ok := cnt.(int64); ok {
|
|
|
|
c = t
|
|
|
|
} else {
|
|
|
|
return keyN
|
|
|
|
}
|
|
|
|
|
|
|
|
ruleFunc, ok := trNLangRules[lang]
|
|
|
|
if !ok {
|
|
|
|
ruleFunc = trNLangRules["en-US"]
|
|
|
|
}
|
|
|
|
|
|
|
|
if ruleFunc(c) == 0 {
|
|
|
|
return key1
|
|
|
|
}
|
|
|
|
return keyN
|
|
|
|
}
|
2019-07-08 10:14:12 +08:00
|
|
|
|
|
|
|
// MigrationIcon returns a Font Awesome name matching the service an issue/comment was migrated from
|
|
|
|
func MigrationIcon(hostname string) string {
|
|
|
|
switch hostname {
|
|
|
|
case "github.com":
|
|
|
|
return "fa-github"
|
|
|
|
default:
|
|
|
|
return "fa-git-alt"
|
|
|
|
}
|
|
|
|
}
|
2019-11-07 21:34:28 +08:00
|
|
|
|
|
|
|
func buildSubjectBodyTemplate(stpl *texttmpl.Template, btpl *template.Template, name string, content []byte) {
|
|
|
|
// Split template into subject and body
|
|
|
|
var subjectContent []byte
|
|
|
|
bodyContent := content
|
|
|
|
loc := mailSubjectSplit.FindIndex(content)
|
|
|
|
if loc != nil {
|
|
|
|
subjectContent = content[0:loc[0]]
|
|
|
|
bodyContent = content[loc[1]:]
|
|
|
|
}
|
|
|
|
if _, err := stpl.New(name).
|
|
|
|
Parse(string(subjectContent)); err != nil {
|
|
|
|
log.Warn("Failed to parse template [%s/subject]: %v", name, err)
|
|
|
|
}
|
|
|
|
if _, err := btpl.New(name).
|
|
|
|
Parse(string(bodyContent)); err != nil {
|
|
|
|
log.Warn("Failed to parse template [%s/body]: %v", name, err)
|
|
|
|
}
|
|
|
|
}
|