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.
2015-11-14 01:10:25 +03:00
package template
2014-04-10 22:20:58 +04:00
import (
"container/list"
"encoding/json"
"fmt"
"html/template"
2016-08-12 02:16:36 +03:00
"mime"
"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
2014-12-22 12:01:52 +03:00
"golang.org/x/net/html/charset"
"golang.org/x/text/transform"
2016-08-12 03:07:09 +03:00
"gopkg.in/editorconfig/editorconfig-core-go.v1"
2014-12-22 12:01:52 +03:00
2015-11-14 01:10:25 +03:00
"github.com/gogits/gogs/models"
"github.com/gogits/gogs/modules/base"
2016-03-26 23:42:20 +03:00
"github.com/gogits/gogs/modules/log"
2016-02-21 01:10:05 +03:00
"github.com/gogits/gogs/modules/markdown"
2015-01-20 08:08:49 +03:00
"github.com/gogits/gogs/modules/setting"
2014-04-10 22:20:58 +04:00
)
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 {
return strings . HasPrefix ( setting . AppUrl , "https" )
} ,
"AppName" : func ( ) string {
return setting . AppName
} ,
"AppSubUrl" : func ( ) string {
return setting . AppSubUrl
} ,
"AppUrl" : func ( ) string {
return setting . AppUrl
} ,
"AppVer" : func ( ) string {
return setting . AppVer
} ,
"AppDomain" : func ( ) string {
return setting . Domain
} ,
"DisableGravatar" : func ( ) bool {
return setting . DisableGravatar
} ,
"LoadTimes" : func ( startTime time . Time ) string {
return fmt . Sprint ( time . Since ( startTime ) . Nanoseconds ( ) / 1e6 ) + "ms"
} ,
"AvatarLink" : base . AvatarLink ,
"Safe" : Safe ,
"Str2html" : Str2html ,
"TimeSince" : base . TimeSince ,
"RawTimeSince" : base . RawTimeSince ,
"FileSize" : base . FileSize ,
"Subtract" : base . Subtract ,
"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" )
} ,
"List" : List ,
"Mail2Domain" : func ( mail string ) string {
if ! strings . Contains ( mail , "@" ) {
return "try.gogs.io"
}
2015-11-14 06:45:33 +03:00
2016-03-07 00:40:04 +03:00
return strings . SplitN ( mail , "@" , 2 ) [ 1 ]
} ,
"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 ]
} ,
"DiffTypeToStr" : DiffTypeToStr ,
"DiffLineTypeToStr" : DiffLineTypeToStr ,
"Sha1" : Sha1 ,
"ShortSha" : base . ShortSha ,
"MD5" : base . EncodeMD5 ,
"ActionContent2Commits" : ActionContent2Commits ,
"EscapePound" : func ( str string ) string {
2016-03-25 04:48:52 +03:00
return strings . NewReplacer ( "%" , "%25" , "#" , "%23" , " " , "%20" ) . Replace ( str )
2016-03-07 00:40:04 +03:00
} ,
"RenderCommitMessage" : RenderCommitMessage ,
"ThemeColorMetaTag" : func ( ) string {
2016-07-23 19:23:54 +03:00
return setting . UI . ThemeColorMetaTag
2016-03-07 00:40:04 +03:00
} ,
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-03-07 00:40:04 +03:00
} }
2015-11-14 06:45:33 +03:00
}
2015-08-08 12:10:34 +03:00
func Safe ( raw string ) template . HTML {
return template . HTML ( raw )
}
2014-04-10 22:20:58 +04:00
func Str2html ( raw string ) template . HTML {
2016-02-21 01:10:05 +03:00
return template . HTML ( markdown . Sanitizer . Sanitize ( raw ) )
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
}
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-08-09 22:56:00 +03:00
func ToUTF8WithErr ( content [ ] byte ) ( error , string ) {
2016-01-01 06:13:47 +03:00
charsetLabel , err := base . DetectEncoding ( content )
if err != nil {
return err , ""
} else if charsetLabel == "UTF-8" {
2014-09-17 08:03:03 +04:00
return nil , string ( content )
}
2014-12-22 12:01:52 +03:00
encoding , _ := charset . Lookup ( charsetLabel )
if encoding == nil {
2015-12-25 13:25:47 +03:00
return fmt . Errorf ( "Unknown encoding: %s" , charsetLabel ) , string ( content )
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
// original left over. This way we won't loose data.
2015-12-25 13:25:47 +03:00
result , n , err := transform . String ( encoding . NewDecoder ( ) , string ( content ) )
2014-12-22 12:01:52 +03:00
if err != nil {
result = result + string ( content [ n : ] )
}
return err , result
2014-09-17 08:03:03 +04:00
}
2016-08-09 22:56:00 +03:00
func ToUTF8 ( content string ) string {
_ , res := ToUTF8WithErr ( [ ] byte ( content ) )
2014-09-17 08:03:03 +04:00
return res
}
2015-09-19 04:57:06 +03:00
// Replaces all prefixes 'old' in 's' with 'new'.
func ReplaceLeft ( s , old , new string ) string {
old_len , new_len , i , n := len ( old ) , len ( new ) , 0 , 0
for ; i < len ( s ) && strings . HasPrefix ( s [ i : ] , old ) ; n += 1 {
i += old_len
}
// simple optimization
if n == 0 {
return s
}
// allocating space for the new string
newLen := n * new_len + len ( s [ i : ] )
replacement := make ( [ ] byte , newLen , newLen )
j := 0
for ; j < n * new_len ; j += new_len {
copy ( replacement [ j : j + new_len ] , new )
}
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.
2015-12-07 02:18:12 +03:00
func RenderCommitMessage ( full bool , msg , urlPrefix string , metas map [ string ] string ) template . HTML {
2015-09-19 04:57:06 +03:00
cleanMsg := template . HTMLEscapeString ( msg )
2016-02-21 01:10:05 +03:00
fullMessage := string ( markdown . RenderIssueIndexPattern ( [ ] byte ( cleanMsg ) , urlPrefix , metas ) )
2015-09-19 04:57:06 +03:00
msgLines := strings . Split ( strings . TrimSpace ( fullMessage ) , "\n" )
2015-12-07 02:18:12 +03:00
numLines := len ( msgLines )
if numLines == 0 {
return template . HTML ( "" )
} else if ! full {
return template . HTML ( msgLines [ 0 ] )
} else if numLines == 1 || ( numLines >= 2 && len ( msgLines [ 1 ] ) == 0 ) {
// First line is a header, standalone or followed by empty line
header := fmt . Sprintf ( "<h3>%s</h3>" , msgLines [ 0 ] )
if numLines >= 2 {
fullMessage = header + fmt . Sprintf ( "\n<pre>%s</pre>" , strings . Join ( msgLines [ 2 : ] , "\n" ) )
} else {
fullMessage = header
}
} else {
// Non-standard git message, there is no header line
fullMessage = fmt . Sprintf ( "<h4>%s</h4>" , strings . Join ( msgLines , "<br>" ) )
2015-09-19 04:57:06 +03:00
}
return template . HTML ( fullMessage )
2015-01-31 02:05:20 +03:00
}
2014-04-10 22:20:58 +04:00
type Actioner interface {
GetOpType ( ) int
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
}
// ActionIcon accepts a int that represents action operation type
// and returns a icon class name.
func ActionIcon ( opType int ) string {
switch opType {
2016-02-22 20:40:00 +03:00
case 1 , 8 : // Create and transfer repository
2014-07-26 08:24:27 +04:00
return "repo"
2015-11-16 19:39:48 +03:00
case 5 , 9 : // Commit repository
2014-07-26 08:24:27 +04:00
return "git-commit"
2016-03-05 20:58:51 +03:00
case 6 : // Create issue
2014-07-26 08:24:27 +04:00
return "issue-opened"
2015-11-16 19:39:48 +03:00
case 7 : // New pull request
return "git-pull-request"
case 10 : // Comment issue
2016-07-16 07:45:13 +03:00
return "comment-discussion"
2015-11-16 19:39:48 +03:00
case 11 : // Merge pull request
return "git-merge"
2016-03-05 20:58:51 +03:00
case 12 , 14 : // Close issue or pull request
2016-02-22 20:40:00 +03:00
return "issue-closed"
2016-03-05 20:58:51 +03:00
case 13 , 15 : // Reopen issue or pull request
return "issue-reopened"
2014-04-10 22:20:58 +04:00
default :
return "invalid type"
}
}
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 {
2016-03-26 23:42:20 +03:00
log . Error ( 4 , "json.Unmarshal:\n%s\nERROR: %v" , act . GetContent ( ) , err )
2014-07-26 08:24:27 +04:00
}
return push
}
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 ]
}
func DiffLineTypeToStr ( diffType int ) string {
switch diffType {
case 2 :
return "add"
case 3 :
return "del"
case 4 :
return "tag"
}
return "same"
}