2018-09-29 11:33:54 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2014-04-10 22:20:58 +04: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-06 20:58:31 +03:00
package templates
2014-04-10 22:20:58 +04:00
import (
2017-03-02 03:25:44 +03:00
"bytes"
2014-04-10 22:20:58 +04:00
"container/list"
"encoding/json"
2017-12-04 02:14:26 +03:00
"errors"
2014-04-10 22:20:58 +04:00
"fmt"
2018-02-27 10:09:18 +03:00
"html"
2014-04-10 22:20:58 +04:00
"html/template"
2016-08-12 02:16:36 +03:00
"mime"
2017-11-28 12:43:51 +03:00
"net/url"
2016-08-12 02:16:36 +03:00
"path/filepath"
2014-05-22 05:37:13 +04:00
"runtime"
2014-04-10 22:20:58 +04:00
"strings"
"time"
2014-05-26 04:11:25 +04:00
2019-05-02 16:09:39 +03:00
"code.gitea.io/gitea/modules/util"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/log"
2017-09-16 20:17:57 +03:00
"code.gitea.io/gitea/modules/markup"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/setting"
2017-11-22 10:09:48 +03:00
"golang.org/x/net/html/charset"
"golang.org/x/text/transform"
"gopkg.in/editorconfig/editorconfig-core-go.v1"
2014-04-10 22:20:58 +04:00
)
2016-11-25 09:23:48 +03:00
// NewFuncMap returns functions for injecting to templates
2016-03-07 00:40:04 +03: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 13:14:25 +03:00
return strings . HasPrefix ( setting . AppURL , "https" )
2016-03-07 00:40:04 +03:00
} ,
"AppName" : func ( ) string {
return setting . AppName
} ,
"AppSubUrl" : func ( ) string {
2016-11-27 13:14:25 +03:00
return setting . AppSubURL
2016-03-07 00:40:04 +03:00
} ,
"AppUrl" : func ( ) string {
2016-11-27 13:14:25 +03:00
return setting . AppURL
2016-03-07 00:40:04 +03:00
} ,
"AppVer" : func ( ) string {
return setting . AppVer
} ,
2017-02-28 03:40:02 +03:00
"AppBuiltWith" : func ( ) string {
2017-03-01 04:45:21 +03:00
return setting . AppBuiltWith
2017-02-28 03:40:02 +03:00
} ,
2016-03-07 00:40:04 +03:00
"AppDomain" : func ( ) string {
return setting . Domain
} ,
"DisableGravatar" : func ( ) bool {
return setting . DisableGravatar
} ,
2016-09-01 08:01:32 +03:00
"ShowFooterTemplateLoadTime" : func ( ) bool {
return setting . ShowFooterTemplateLoadTime
} ,
2016-03-07 00:40:04 +03:00
"LoadTimes" : func ( startTime time . Time ) string {
return fmt . Sprint ( time . Since ( startTime ) . Nanoseconds ( ) / 1e6 ) + "ms"
} ,
2017-12-11 07:37:04 +03:00
"AvatarLink" : base . AvatarLink ,
"Safe" : Safe ,
"SafeJS" : SafeJS ,
"Str2html" : Str2html ,
"TimeSince" : base . TimeSince ,
"TimeSinceUnix" : base . TimeSinceUnix ,
"RawTimeSince" : base . RawTimeSince ,
"FileSize" : base . FileSize ,
"Subtract" : base . Subtract ,
2018-05-01 10:04:36 +03:00
"EntryIcon" : base . EntryIcon ,
2016-03-07 00:40:04 +03: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" )
} ,
2017-04-11 16:30:15 +03:00
"SizeFmt" : func ( s int64 ) string {
return base . FileSize ( s )
} ,
2016-03-07 00:40:04 +03:00
"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 ]
} ,
2018-08-29 16:43:58 +03:00
"EllipsisString" : base . EllipsisString ,
"DiffTypeToStr" : DiffTypeToStr ,
"DiffLineTypeToStr" : DiffLineTypeToStr ,
"Sha1" : Sha1 ,
"ShortSha" : base . ShortSha ,
"MD5" : base . EncodeMD5 ,
2016-03-07 00:40:04 +03:00
"ActionContent2Commits" : ActionContent2Commits ,
2017-11-28 12:43:51 +03:00
"PathEscape" : url . PathEscape ,
2016-03-07 00:40:04 +03:00
"EscapePound" : func ( str string ) string {
2016-09-18 18:46:52 +03:00
return strings . NewReplacer ( "%" , "%25" , "#" , "%23" , " " , "%20" , "?" , "%3F" ) . Replace ( str )
2016-03-07 00:40:04 +03:00
} ,
2019-05-02 16:09:39 +03:00
"PathEscapeSegments" : util . PathEscapeSegments ,
"URLJoin" : util . URLJoin ,
2017-11-30 08:08:40 +03:00
"RenderCommitMessage" : RenderCommitMessage ,
"RenderCommitMessageLink" : RenderCommitMessageLink ,
"RenderCommitBody" : RenderCommitBody ,
"IsMultilineCommitMessage" : IsMultilineCommitMessage ,
2016-03-07 00:40:04 +03:00
"ThemeColorMetaTag" : func ( ) string {
2016-07-23 19:23:54 +03:00
return setting . UI . ThemeColorMetaTag
2016-03-07 00:40:04 +03:00
} ,
2017-04-01 04:03:01 +03: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
} ,
2016-08-12 02:16:36 +03:00
"FilenameIsImage" : func ( filename string ) bool {
mimeType := mime . TypeByExtension ( filepath . Ext ( filename ) )
return strings . HasPrefix ( mimeType , "image/" )
} ,
2016-08-12 03:07:09 +03:00
"TabSizeClass" : func ( ec * editorconfig . Editorconfig , filename string ) string {
if ec != nil {
def := ec . GetDefinitionForFilename ( filename )
if def . TabWidth > 0 {
return fmt . Sprintf ( "tab-size-%d" , def . TabWidth )
}
}
return "tab-size-8"
} ,
2016-12-28 19:35:52 +03:00
"SubJumpablePath" : func ( str string ) [ ] string {
var path [ ] string
index := strings . LastIndex ( str , "/" )
if index != - 1 && index != len ( str ) {
2017-01-27 18:03:32 +03:00
path = append ( path , str [ 0 : index + 1 ] )
path = append ( path , str [ index + 1 : ] )
2016-12-28 19:35:52 +03:00
} else {
path = append ( path , str )
}
return path
} ,
2017-03-02 03:25:44 +03: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 12:25:42 +03:00
"DisableGitHooks" : func ( ) bool {
return setting . DisableGitHooks
} ,
2018-08-24 08:00:22 +03:00
"DisableImportLocal" : func ( ) bool {
return ! setting . ImportLocalPaths
} ,
2017-10-15 02:17:39 +03:00
"TrN" : TrN ,
2017-12-04 02:14:26 +03: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 08:58:47 +03:00
"Printf" : fmt . Sprintf ,
"Escape" : Escape ,
"Sec2Time" : models . SecToTime ,
2018-05-01 22:05:28 +03:00
"ParseDeadline" : func ( deadline string ) [ ] string {
return strings . Split ( deadline , "|" )
} ,
2018-07-06 00:25:04 +03:00
"DefaultTheme" : func ( ) string {
return setting . UI . DefaultTheme
} ,
2018-08-06 07:43:22 +03: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
} ,
2016-03-07 00:40:04 +03:00
} }
2015-11-14 06:45:33 +03:00
}
2016-11-25 09:23:48 +03:00
// Safe render raw as HTML
2015-08-08 12:10:34 +03:00
func Safe ( raw string ) template . HTML {
return template . HTML ( raw )
}
2017-08-23 17:58:05 +03:00
// SafeJS renders raw as JS
func SafeJS ( raw string ) template . JS {
return template . JS ( raw )
}
2016-11-25 09:23:48 +03:00
// Str2html render Markdown text to HTML
2014-04-10 22:20:58 +04:00
func Str2html ( raw string ) template . HTML {
2017-09-16 20:17:57 +03:00
return template . HTML ( markup . Sanitize ( raw ) )
2014-04-10 22:20:58 +04:00
}
2018-02-11 16:42:28 +03:00
// Escape escapes a HTML string
func Escape ( raw string ) string {
return html . EscapeString ( raw )
}
2016-11-25 09:23:48 +03:00
// List traversings the list
2014-04-10 22:20:58 +04: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 09:23:48 +03:00
// Sha1 returns sha1 sum of string
2015-02-19 00:52:22 +03:00
func Sha1 ( str string ) string {
2015-11-14 01:10:25 +03:00
return base . EncodeSha1 ( str )
2014-12-09 10:18:25 +03:00
}
2016-11-25 09:23:48 +03:00
// ToUTF8WithErr converts content to UTF8 encoding
func ToUTF8WithErr ( content [ ] byte ) ( string , error ) {
2016-01-01 06:13:47 +03:00
charsetLabel , err := base . DetectEncoding ( content )
if err != nil {
2016-11-25 09:23:48 +03:00
return "" , err
2016-01-01 06:13:47 +03:00
} else if charsetLabel == "UTF-8" {
2019-04-26 15:00:30 +03:00
return string ( base . RemoveBOMIfPresent ( content ) ) , nil
2014-09-17 08:03:03 +04:00
}
2014-12-22 12:01:52 +03:00
encoding , _ := charset . Lookup ( charsetLabel )
if encoding == nil {
2016-11-25 09:23:48 +03:00
return string ( content ) , fmt . Errorf ( "Unknown encoding: %s" , charsetLabel )
2014-09-17 08:03:03 +04:00
}
2014-12-22 12:01:52 +03:00
// If there is an error, we concatenate the nicely decoded part and the
2018-09-29 11:33:54 +03:00
// original left over. This way we won't lose data.
2019-04-26 15:00:30 +03:00
result , n , err := transform . Bytes ( encoding . NewDecoder ( ) , content )
2014-12-22 12:01:52 +03:00
if err != nil {
2019-04-26 15:00:30 +03:00
result = append ( result , content [ n : ] ... )
2014-12-22 12:01:52 +03:00
}
2019-04-26 15:00:30 +03:00
result = base . RemoveBOMIfPresent ( result )
return string ( result ) , err
2014-09-17 08:03:03 +04:00
}
2018-09-29 11:33:54 +03:00
// ToUTF8WithFallback detects the encoding of content and coverts to UTF-8 if possible
func ToUTF8WithFallback ( content [ ] byte ) [ ] byte {
charsetLabel , err := base . DetectEncoding ( content )
if err != nil || charsetLabel == "UTF-8" {
2019-04-26 15:00:30 +03:00
return base . RemoveBOMIfPresent ( content )
2018-09-29 11:33:54 +03:00
}
encoding , _ := charset . Lookup ( charsetLabel )
if encoding == nil {
return content
}
// If there is an error, we concatenate the nicely decoded part and the
// original left over. This way we won't lose data.
result , n , err := transform . Bytes ( encoding . NewDecoder ( ) , content )
if err != nil {
return append ( result , content [ n : ] ... )
}
2019-04-26 15:00:30 +03:00
return base . RemoveBOMIfPresent ( result )
2018-09-29 11:33:54 +03:00
}
2016-11-25 09:23:48 +03:00
// ToUTF8 converts content to UTF8 encoding and ignore error
2016-08-09 22:56:00 +03:00
func ToUTF8 ( content string ) string {
2016-11-25 09:23:48 +03:00
res , _ := ToUTF8WithErr ( [ ] byte ( content ) )
2014-09-17 08:03:03 +04:00
return res
}
2016-11-25 09:23:48 +03:00
// ReplaceLeft replaces all prefixes 'old' in 's' with 'new'.
2015-09-19 04:57:06 +03:00
func ReplaceLeft ( s , old , new string ) string {
2016-11-25 09:23:48 +03:00
oldLen , newLen , i , n := len ( old ) , len ( new ) , 0 , 0
for ; i < len ( s ) && strings . HasPrefix ( s [ i : ] , old ) ; n ++ {
i += oldLen
2015-09-19 04:57:06 +03:00
}
// simple optimization
if n == 0 {
return s
}
// allocating space for the new string
2016-11-25 09:23:48 +03:00
curLen := n * newLen + len ( s [ i : ] )
replacement := make ( [ ] byte , curLen , curLen )
2015-09-19 04:57:06 +03:00
j := 0
2016-11-25 09:23:48 +03:00
for ; j < n * newLen ; j += newLen {
copy ( replacement [ j : j + newLen ] , new )
2015-09-19 04:57:06 +03:00
}
copy ( replacement [ j : ] , s [ i : ] )
return string ( replacement )
}
2015-01-31 02:05:20 +03:00
// RenderCommitMessage renders commit message with XSS-safe and special links.
2017-11-13 04:35:55 +03:00
func RenderCommitMessage ( msg , urlPrefix string , metas map [ string ] string ) template . HTML {
2018-02-27 10:09:18 +03:00
return RenderCommitMessageLink ( msg , urlPrefix , "" , metas )
2017-11-13 04:35:55 +03:00
}
// RenderCommitMessageLink renders commit message as a XXS-safe link to the provided
// default url, handling for special links.
2018-02-27 10:09:18 +03:00
func RenderCommitMessageLink ( msg , urlPrefix , urlDefault string , metas map [ string ] string ) template . HTML {
2015-09-19 04:57:06 +03:00
cleanMsg := template . HTMLEscapeString ( msg )
2018-02-27 10:09:18 +03: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 10:48:31 +03:00
log . Error ( "RenderCommitMessage: %v" , err )
2018-02-27 10:09:18 +03:00
return ""
}
msgLines := strings . Split ( strings . TrimSpace ( string ( fullMessage ) ) , "\n" )
2017-11-13 04:35:55 +03:00
if len ( msgLines ) == 0 {
2015-12-07 02:18:12 +03:00
return template . HTML ( "" )
2015-09-19 04:57:06 +03:00
}
2017-11-13 04:35:55 +03:00
return template . HTML ( msgLines [ 0 ] )
2015-01-31 02:05:20 +03:00
}
2017-11-30 08:08:40 +03:00
// RenderCommitBody extracts the body of a commit message without its title.
func RenderCommitBody ( msg , urlPrefix string , metas map [ string ] string ) template . HTML {
cleanMsg := template . HTMLEscapeString ( msg )
2018-02-27 10:09:18 +03:00
fullMessage , err := markup . RenderCommitMessage ( [ ] byte ( cleanMsg ) , urlPrefix , "" , metas )
if err != nil {
2019-04-02 10:48:31 +03:00
log . Error ( "RenderCommitMessage: %v" , err )
2018-02-27 10:09:18 +03:00
return ""
}
body := strings . Split ( strings . TrimSpace ( string ( fullMessage ) ) , "\n" )
2017-11-30 08:08:40 +03:00
if len ( body ) == 0 {
return template . HTML ( "" )
}
return template . HTML ( strings . Join ( body [ 1 : ] , "\n" ) )
}
// IsMultilineCommitMessage checks to see if a commit message contains multiple lines.
func IsMultilineCommitMessage ( msg string ) bool {
2018-06-15 17:07:48 +03:00
return strings . Count ( strings . TrimSpace ( msg ) , "\n" ) >= 1
2017-11-30 08:08:40 +03:00
}
2016-11-25 09:23:48 +03:00
// Actioner describes an action
2014-04-10 22:20:58 +04:00
type Actioner interface {
2017-09-20 04:22:42 +03:00
GetOpType ( ) models . ActionType
2014-04-10 22:20:58 +04:00
GetActUserName ( ) string
2014-05-09 10:42:50 +04:00
GetRepoUserName ( ) string
2014-04-10 22:20:58 +04:00
GetRepoName ( ) string
2015-09-01 16:29:52 +03:00
GetRepoPath ( ) string
GetRepoLink ( ) string
2014-04-10 22:20:58 +04:00
GetBranch ( ) string
GetContent ( ) string
2015-09-01 16:29:52 +03:00
GetCreate ( ) time . Time
GetIssueInfos ( ) [ ] string
2014-04-10 22:20:58 +04:00
}
2017-09-20 04:22:42 +03:00
// ActionIcon accepts an action operation type and returns an icon class name.
func ActionIcon ( opType models . ActionType ) string {
2014-04-10 22:20:58 +04:00
switch opType {
2017-09-20 04:22:42 +03:00
case models . ActionCreateRepo , models . ActionTransferRepo :
2014-07-26 08:24:27 +04:00
return "repo"
2017-09-21 10:43:26 +03:00
case models . ActionCommitRepo , models . ActionPushTag , models . ActionDeleteTag , models . ActionDeleteBranch :
2014-07-26 08:24:27 +04:00
return "git-commit"
2017-09-20 04:22:42 +03:00
case models . ActionCreateIssue :
2014-07-26 08:24:27 +04:00
return "issue-opened"
2017-09-20 04:22:42 +03:00
case models . ActionCreatePullRequest :
2015-11-16 19:39:48 +03:00
return "git-pull-request"
2017-09-20 04:22:42 +03:00
case models . ActionCommentIssue :
2016-07-16 07:45:13 +03:00
return "comment-discussion"
2017-09-20 04:22:42 +03:00
case models . ActionMergePullRequest :
2015-11-16 19:39:48 +03:00
return "git-merge"
2017-09-20 04:22:42 +03:00
case models . ActionCloseIssue , models . ActionClosePullRequest :
2016-02-22 20:40:00 +03:00
return "issue-closed"
2017-09-20 04:22:42 +03:00
case models . ActionReopenIssue , models . ActionReopenPullRequest :
2016-03-05 20:58:51 +03:00
return "issue-reopened"
2018-09-07 05:06:09 +03:00
case models . ActionMirrorSyncPush , models . ActionMirrorSyncCreate , models . ActionMirrorSyncDelete :
return "repo-clone"
2014-04-10 22:20:58 +04:00
default :
return "invalid type"
}
}
2016-11-25 09:23:48 +03:00
// ActionContent2Commits converts action content to push commits
2015-11-14 01:10:25 +03:00
func ActionContent2Commits ( act Actioner ) * models . PushCommits {
push := models . NewPushCommits ( )
if err := json . Unmarshal ( [ ] byte ( act . GetContent ( ) ) , push ) ; err != nil {
2019-04-02 10:48:31 +03:00
log . Error ( "json.Unmarshal:\n%s\nERROR: %v" , act . GetContent ( ) , err )
2014-07-26 08:24:27 +04:00
}
return push
}
2016-11-25 09:23:48 +03:00
// DiffTypeToStr returns diff type name
2014-04-10 22:20:58 +04:00
func DiffTypeToStr ( diffType int ) string {
diffTypes := map [ int ] string {
2015-11-03 17:52:17 +03:00
1 : "add" , 2 : "modify" , 3 : "del" , 4 : "rename" ,
2014-04-10 22:20:58 +04:00
}
return diffTypes [ diffType ]
}
2016-11-25 09:23:48 +03:00
// DiffLineTypeToStr returns diff line type name
2014-04-10 22:20:58 +04: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 02:17:39 +03: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 01:44:43 +03:00
"fr-FR" : func ( cnt int64 ) int {
if cnt > - 2 && cnt < 2 {
return 0
}
return 1
} ,
2017-10-15 02:17:39 +03: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
}