2019-04-20 04:47:00 +02:00
// Copyright 2019 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 repo
import (
"bytes"
"container/list"
"fmt"
"html"
gotemplate "html/template"
2021-04-05 17:30:52 +02:00
"net/http"
2019-04-20 04:47:00 +02:00
"strings"
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/context"
"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/highlight"
"code.gitea.io/gitea/modules/log"
2020-12-03 19:46:11 +01:00
"code.gitea.io/gitea/modules/templates"
2019-08-15 22:46:21 +08:00
"code.gitea.io/gitea/modules/timeutil"
2019-04-20 04:47:00 +02:00
)
const (
tplBlame base . TplName = "repo/home"
)
// RefBlame render blame page
func RefBlame ( ctx * context . Context ) {
fileName := ctx . Repo . TreePath
if len ( fileName ) == 0 {
ctx . NotFound ( "Blame FileName" , nil )
return
}
userName := ctx . Repo . Owner . Name
repoName := ctx . Repo . Repository . Name
commitID := ctx . Repo . CommitID
commit , err := ctx . Repo . GitRepo . GetCommit ( commitID )
if err != nil {
if git . IsErrNotExist ( err ) {
ctx . NotFound ( "Repo.GitRepo.GetCommit" , err )
} else {
ctx . ServerError ( "Repo.GitRepo.GetCommit" , err )
}
return
}
if len ( commitID ) != 40 {
commitID = commit . ID . String ( )
}
branchLink := ctx . Repo . RepoLink + "/src/" + ctx . Repo . BranchNameSubURL ( )
treeLink := branchLink
rawLink := ctx . Repo . RepoLink + "/raw/" + ctx . Repo . BranchNameSubURL ( )
if len ( ctx . Repo . TreePath ) > 0 {
treeLink += "/" + ctx . Repo . TreePath
}
var treeNames [ ] string
paths := make ( [ ] string , 0 , 5 )
if len ( ctx . Repo . TreePath ) > 0 {
treeNames = strings . Split ( ctx . Repo . TreePath , "/" )
for i := range treeNames {
paths = append ( paths , strings . Join ( treeNames [ : i + 1 ] , "/" ) )
}
ctx . Data [ "HasParentPath" ] = true
if len ( paths ) - 2 >= 0 {
ctx . Data [ "ParentPath" ] = "/" + paths [ len ( paths ) - 1 ]
}
}
// Show latest commit info of repository in table header,
// or of directory if not in root directory.
latestCommit := ctx . Repo . Commit
if len ( ctx . Repo . TreePath ) > 0 {
latestCommit , err = ctx . Repo . Commit . GetCommitByPath ( ctx . Repo . TreePath )
if err != nil {
ctx . ServerError ( "GetCommitByPath" , err )
return
}
}
ctx . Data [ "LatestCommit" ] = latestCommit
ctx . Data [ "LatestCommitVerification" ] = models . ParseCommitWithSignature ( latestCommit )
ctx . Data [ "LatestCommitUser" ] = models . ValidateCommitWithEmail ( latestCommit )
2020-12-18 03:33:32 +00:00
statuses , err := models . GetLatestCommitStatus ( ctx . Repo . Repository . ID , ctx . Repo . Commit . ID . String ( ) , models . ListOptions { } )
2019-04-20 04:47:00 +02:00
if err != nil {
log . Error ( "GetLatestCommitStatus: %v" , err )
}
// Get current entry user currently looking at.
entry , err := ctx . Repo . Commit . GetTreeEntryByPath ( ctx . Repo . TreePath )
if err != nil {
ctx . NotFoundOrServerError ( "Repo.Commit.GetTreeEntryByPath" , git . IsErrNotExist , err )
return
}
blob := entry . Blob ( )
ctx . Data [ "LatestCommitStatus" ] = models . CalcCommitStatus ( statuses )
2020-12-20 04:13:12 +01:00
ctx . Data [ "LatestCommitStatuses" ] = statuses
2019-04-20 04:47:00 +02:00
ctx . Data [ "Paths" ] = paths
ctx . Data [ "TreeLink" ] = treeLink
ctx . Data [ "TreeNames" ] = treeNames
ctx . Data [ "BranchLink" ] = branchLink
2020-06-30 17:34:03 -04:00
2019-04-20 04:47:00 +02:00
ctx . Data [ "RawFileLink" ] = rawLink + "/" + ctx . Repo . TreePath
ctx . Data [ "PageIsViewCode" ] = true
ctx . Data [ "IsBlame" ] = true
ctx . Data [ "FileSize" ] = blob . Size ( )
ctx . Data [ "FileName" ] = blob . Name ( )
2020-06-15 20:39:39 +02:00
ctx . Data [ "NumLines" ] , err = blob . GetBlobLineCount ( )
if err != nil {
ctx . NotFound ( "GetBlobLineCount" , err )
return
}
2021-05-31 07:18:11 +01:00
blameReader , err := git . CreateBlameReader ( ctx , models . RepoPath ( userName , repoName ) , commitID , fileName )
2019-04-20 04:47:00 +02:00
if err != nil {
ctx . NotFound ( "CreateBlameReader" , err )
return
}
defer blameReader . Close ( )
2019-06-27 02:15:26 +08:00
blameParts := make ( [ ] git . BlamePart , 0 )
2019-04-20 04:47:00 +02:00
for {
blamePart , err := blameReader . NextPart ( )
if err != nil {
ctx . NotFound ( "NextPart" , err )
return
}
if blamePart == nil {
break
}
blameParts = append ( blameParts , * blamePart )
}
commitNames := make ( map [ string ] models . UserCommit )
commits := list . New ( )
for _ , part := range blameParts {
sha := part . Sha
if _ , ok := commitNames [ sha ] ; ok {
continue
}
commit , err := ctx . Repo . GitRepo . GetCommit ( sha )
if err != nil {
if git . IsErrNotExist ( err ) {
ctx . NotFound ( "Repo.GitRepo.GetCommit" , err )
} else {
ctx . ServerError ( "Repo.GitRepo.GetCommit" , err )
}
return
}
commits . PushBack ( commit )
commitNames [ commit . ID . String ( ) ] = models . UserCommit { }
}
commits = models . ValidateCommitsWithEmails ( commits )
for e := commits . Front ( ) ; e != nil ; e = e . Next ( ) {
c := e . Value . ( models . UserCommit )
commitNames [ c . ID . String ( ) ] = c
}
2020-05-05 23:51:49 +02:00
// Get Topics of this repo
renderRepoTopics ( ctx )
if ctx . Written ( ) {
return
}
2019-04-20 04:47:00 +02:00
renderBlame ( ctx , blameParts , commitNames )
2021-04-05 17:30:52 +02:00
ctx . HTML ( http . StatusOK , tplBlame )
2019-04-20 04:47:00 +02:00
}
2019-06-27 02:15:26 +08:00
func renderBlame ( ctx * context . Context , blameParts [ ] git . BlamePart , commitNames map [ string ] models . UserCommit ) {
2019-04-20 04:47:00 +02:00
repoLink := ctx . Repo . RepoLink
2019-06-12 21:41:28 +02:00
var lines = make ( [ ] string , 0 )
2019-04-20 04:47:00 +02:00
var commitInfo bytes . Buffer
var lineNumbers bytes . Buffer
var codeLines bytes . Buffer
var i = 0
for pi , part := range blameParts {
for index , line := range part . Lines {
i ++
lines = append ( lines , line )
var attr = ""
if len ( part . Lines ) - 1 == index && len ( blameParts ) - 1 != pi {
attr = " bottom-line"
}
commit := commitNames [ part . Sha ]
if index == 0 {
// User avatar image
2019-08-15 22:46:21 +08:00
commitSince := timeutil . TimeSinceUnix ( timeutil . TimeStamp ( commit . Author . When . Unix ( ) ) , ctx . Data [ "Lang" ] . ( string ) )
2020-12-03 19:46:11 +01:00
var avatar string
2019-04-20 04:47:00 +02:00
if commit . User != nil {
2020-12-03 19:46:11 +01:00
avatar = string ( templates . Avatar ( commit . User , 18 , "mr-3" ) )
2019-04-20 04:47:00 +02:00
} else {
2020-12-03 19:46:11 +01:00
avatar = string ( templates . AvatarByEmail ( commit . Author . Email , commit . Author . Name , 18 , "mr-3" ) )
2019-04-20 04:47:00 +02:00
}
2020-12-03 19:46:11 +01:00
2019-04-20 04:47:00 +02:00
commitInfo . WriteString ( fmt . Sprintf ( ` <div class="blame-info%s"><div class="blame-data"><div class="blame-avatar">%s</div><div class="blame-message"><a href="%s/commit/%s" title="%[5]s">%[5]s</a></div><div class="blame-time">%s</div></div></div> ` , attr , avatar , repoLink , part . Sha , html . EscapeString ( commit . CommitMessage ) , commitSince ) )
} else {
commitInfo . WriteString ( fmt . Sprintf ( ` <div class="blame-info%s">​</div> ` , attr ) )
}
//Line number
if len ( part . Lines ) - 1 == index && len ( blameParts ) - 1 != pi {
2020-06-15 20:39:39 +02:00
lineNumbers . WriteString ( fmt . Sprintf ( ` <span id="L%d" data-line-number="%d" class="bottom-line"></span> ` , i , i ) )
2019-04-20 04:47:00 +02:00
} else {
2020-06-15 20:39:39 +02:00
lineNumbers . WriteString ( fmt . Sprintf ( ` <span id="L%d" data-line-number="%d"></span> ` , i , i ) )
2019-04-20 04:47:00 +02:00
}
2019-05-11 23:27:39 +03:00
if i != len ( lines ) - 1 {
2019-04-20 04:47:00 +02:00
line += "\n"
}
2020-06-30 17:34:03 -04:00
fileName := fmt . Sprintf ( "%v" , ctx . Data [ "FileName" ] )
line = highlight . Code ( fileName , line )
2020-11-04 08:14:07 +01:00
line = ` <code class="code-inner"> ` + line + ` </code> `
2019-04-20 04:47:00 +02:00
if len ( part . Lines ) - 1 == index && len ( blameParts ) - 1 != pi {
codeLines . WriteString ( fmt . Sprintf ( ` <li class="L%d bottom-line" rel="L%d">%s</li> ` , i , i , line ) )
} else {
codeLines . WriteString ( fmt . Sprintf ( ` <li class="L%d" rel="L%d">%s</li> ` , i , i , line ) )
}
}
}
ctx . Data [ "BlameContent" ] = gotemplate . HTML ( codeLines . String ( ) )
ctx . Data [ "BlameCommitInfo" ] = gotemplate . HTML ( commitInfo . String ( ) )
ctx . Data [ "BlameLineNums" ] = gotemplate . HTML ( lineNumbers . String ( ) )
}