2017-09-14 09:51:32 +03:00
// Copyright 2017 The Gitea Authors. All rights reserved.
2014-07-26 08:24:27 +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.
package repo
import (
"bytes"
2021-10-08 16:08:22 +03:00
gocontext "context"
2017-01-29 23:13:57 +03:00
"encoding/base64"
2016-08-09 22:56:00 +03:00
"fmt"
gotemplate "html/template"
2021-01-22 20:49:13 +03:00
"io"
2021-04-05 18:30:52 +03:00
"net/http"
2019-10-13 16:23:14 +03:00
"net/url"
2014-07-26 08:24:27 +04:00
"path"
2020-07-01 00:34:03 +03:00
"strconv"
2014-07-26 08:24:27 +04:00
"strings"
2021-10-08 16:08:22 +03:00
"time"
2014-07-26 08:24:27 +04:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/models"
2021-12-10 11:14:24 +03:00
asymkey_model "code.gitea.io/gitea/models/asymkey"
2021-09-24 14:32:56 +03:00
"code.gitea.io/gitea/models/db"
2021-12-10 04:27:50 +03:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-09 22:57:58 +03:00
unit_model "code.gitea.io/gitea/models/unit"
2021-11-24 12:49:20 +03:00
user_model "code.gitea.io/gitea/models/user"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/base"
2020-02-01 22:11:32 +03:00
"code.gitea.io/gitea/modules/cache"
2019-08-15 15:07:28 +03:00
"code.gitea.io/gitea/modules/charset"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/context"
2019-03-27 12:33:00 +03:00
"code.gitea.io/gitea/modules/git"
2016-12-06 20:58:31 +03:00
"code.gitea.io/gitea/modules/highlight"
2016-12-26 04:16:37 +03:00
"code.gitea.io/gitea/modules/lfs"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/log"
2017-04-21 10:01:08 +03:00
"code.gitea.io/gitea/modules/markup"
2022-06-06 11:01:49 +03:00
repo_module "code.gitea.io/gitea/modules/repository"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/setting"
2021-06-30 22:14:53 +03:00
"code.gitea.io/gitea/modules/structs"
2021-06-05 15:32:19 +03:00
"code.gitea.io/gitea/modules/typesniffer"
2021-10-25 00:12:43 +03:00
"code.gitea.io/gitea/modules/util"
2022-03-13 19:40:47 +03:00
"code.gitea.io/gitea/routers/web/feed"
2014-07-26 08:24:27 +04:00
)
const (
2021-10-08 16:08:22 +03:00
tplRepoEMPTY base . TplName = "repo/empty"
tplRepoHome base . TplName = "repo/home"
tplRepoViewList base . TplName = "repo/view_list"
tplWatchers base . TplName = "repo/watchers"
tplForks base . TplName = "repo/forks"
tplMigrating base . TplName = "repo/migrate/migrating"
2014-07-26 08:24:27 +04:00
)
2020-02-22 02:04:20 +03:00
type namedBlob struct {
name string
isSymlink bool
blob * git . Blob
}
2020-07-01 00:34:03 +03:00
func linesBytesCount ( s [ ] byte ) int {
nl := [ ] byte { '\n' }
n := bytes . Count ( s , nl )
if len ( s ) > 0 && ! bytes . HasSuffix ( s , nl ) {
n ++
}
return n
}
2020-02-22 02:04:20 +03:00
// FIXME: There has to be a more efficient way of doing this
func getReadmeFileFromPath ( commit * git . Commit , treePath string ) ( * namedBlob , error ) {
tree , err := commit . SubTree ( treePath )
if err != nil {
return nil , err
}
entries , err := tree . ListEntries ( )
if err != nil {
return nil , err
}
var readmeFiles [ 4 ] * namedBlob
2022-01-20 20:46:10 +03:00
exts := [ ] string { ".md" , ".txt" , "" } // sorted by priority
2020-02-22 02:04:20 +03:00
for _ , entry := range entries {
if entry . IsDir ( ) {
continue
}
for i , ext := range exts {
if markup . IsReadmeFile ( entry . Name ( ) , ext ) {
if readmeFiles [ i ] == nil || base . NaturalSortLess ( readmeFiles [ i ] . name , entry . Blob ( ) . Name ( ) ) {
name := entry . Name ( )
isSymlink := entry . IsLink ( )
target := entry
if isSymlink {
target , err = entry . FollowLinks ( )
if err != nil && ! git . IsErrBadLink ( err ) {
return nil , err
}
}
if target != nil && ( target . IsExecutable ( ) || target . IsRegular ( ) ) {
readmeFiles [ i ] = & namedBlob {
name ,
isSymlink ,
target . Blob ( ) ,
}
}
}
}
}
if markup . IsReadmeFile ( entry . Name ( ) ) {
if readmeFiles [ 3 ] == nil || base . NaturalSortLess ( readmeFiles [ 3 ] . name , entry . Blob ( ) . Name ( ) ) {
name := entry . Name ( )
isSymlink := entry . IsLink ( )
if isSymlink {
entry , err = entry . FollowLinks ( )
if err != nil && ! git . IsErrBadLink ( err ) {
return nil , err
}
}
if entry != nil && ( entry . IsExecutable ( ) || entry . IsRegular ( ) ) {
readmeFiles [ 3 ] = & namedBlob {
name ,
isSymlink ,
entry . Blob ( ) ,
}
}
}
}
}
var readmeFile * namedBlob
for _ , f := range readmeFiles {
if f != nil {
readmeFile = f
break
}
}
return readmeFile , nil
}
2016-08-30 12:08:38 +03:00
func renderDirectory ( ctx * context . Context , treeLink string ) {
2021-10-08 16:08:22 +03:00
entries := renderDirectoryFiles ( ctx , 1 * time . Second )
if ctx . Written ( ) {
2014-07-26 08:24:27 +04:00
return
}
2021-12-15 10:50:11 +03:00
if ctx . Repo . TreePath != "" {
ctx . Data [ "Title" ] = ctx . Tr ( "repo.file.title" , ctx . Repo . Repository . Name + "/" + path . Base ( ctx . Repo . TreePath ) , ctx . Repo . RefName )
}
2022-04-26 23:31:15 +03:00
// Check permission to add or upload new file.
if ctx . Repo . CanWrite ( unit_model . TypeCode ) && ctx . Repo . IsViewBranch {
ctx . Data [ "CanAddFile" ] = ! ctx . Repo . Repository . IsArchived
ctx . Data [ "CanUploadFile" ] = setting . Repository . Upload . Enabled && ! ctx . Repo . Repository . IsArchived
}
readmeFile , readmeTreelink := findReadmeFile ( ctx , entries , treeLink )
if ctx . Written ( ) || readmeFile == nil {
return
}
renderReadmeFile ( ctx , readmeFile , readmeTreelink )
}
func findReadmeFile ( ctx * context . Context , entries git . Entries , treeLink string ) ( * namedBlob , string ) {
2019-01-14 22:15:06 +03:00
// 3 for the extensions in exts[] in order
// the last one is for a readme that doesn't
// strictly match an extension
2020-02-22 02:04:20 +03:00
var readmeFiles [ 4 ] * namedBlob
var docsEntries [ 3 ] * git . TreeEntry
2022-01-20 20:46:10 +03:00
exts := [ ] string { ".md" , ".txt" , "" } // sorted by priority
2016-08-30 12:08:38 +03:00
for _ , entry := range entries {
2017-05-02 11:57:54 +03:00
if entry . IsDir ( ) {
2020-02-22 02:04:20 +03:00
lowerName := strings . ToLower ( entry . Name ( ) )
switch lowerName {
case "docs" :
if entry . Name ( ) == "docs" || docsEntries [ 0 ] == nil {
docsEntries [ 0 ] = entry
}
case ".gitea" :
if entry . Name ( ) == ".gitea" || docsEntries [ 1 ] == nil {
docsEntries [ 1 ] = entry
}
case ".github" :
if entry . Name ( ) == ".github" || docsEntries [ 2 ] == nil {
docsEntries [ 2 ] = entry
}
}
2017-05-02 11:57:54 +03:00
continue
}
2019-01-14 22:15:06 +03:00
for i , ext := range exts {
if markup . IsReadmeFile ( entry . Name ( ) , ext ) {
2020-02-22 02:04:20 +03:00
log . Debug ( "%s" , entry . Name ( ) )
name := entry . Name ( )
isSymlink := entry . IsLink ( )
target := entry
if isSymlink {
2021-10-08 16:08:22 +03:00
var err error
2020-02-22 02:04:20 +03:00
target , err = entry . FollowLinks ( )
if err != nil && ! git . IsErrBadLink ( err ) {
ctx . ServerError ( "FollowLinks" , err )
2022-04-26 23:31:15 +03:00
return nil , ""
2020-02-22 02:04:20 +03:00
}
}
log . Debug ( "%t" , target == nil )
if target != nil && ( target . IsExecutable ( ) || target . IsRegular ( ) ) {
readmeFiles [ i ] = & namedBlob {
name ,
isSymlink ,
target . Blob ( ) ,
}
}
2019-01-14 22:15:06 +03:00
}
}
if markup . IsReadmeFile ( entry . Name ( ) ) {
2020-02-22 02:04:20 +03:00
name := entry . Name ( )
isSymlink := entry . IsLink ( )
if isSymlink {
2021-10-08 16:08:22 +03:00
var err error
2020-02-22 02:04:20 +03:00
entry , err = entry . FollowLinks ( )
if err != nil && ! git . IsErrBadLink ( err ) {
ctx . ServerError ( "FollowLinks" , err )
2022-04-26 23:31:15 +03:00
return nil , ""
2020-02-22 02:04:20 +03:00
}
}
if entry != nil && ( entry . IsExecutable ( ) || entry . IsRegular ( ) ) {
readmeFiles [ 3 ] = & namedBlob {
name ,
isSymlink ,
entry . Blob ( ) ,
}
}
2016-08-30 12:08:38 +03:00
}
2019-01-14 22:15:06 +03:00
}
2016-08-30 12:08:38 +03:00
2020-02-22 02:04:20 +03:00
var readmeFile * namedBlob
readmeTreelink := treeLink
2019-01-14 22:15:06 +03:00
for _ , f := range readmeFiles {
if f != nil {
readmeFile = f
2017-05-02 11:57:54 +03:00
break
}
2016-08-30 12:08:38 +03:00
}
2020-02-22 02:04:20 +03:00
if ctx . Repo . TreePath == "" && readmeFile == nil {
for _ , entry := range docsEntries {
if entry == nil {
continue
}
2021-10-08 16:08:22 +03:00
var err error
2020-02-22 02:04:20 +03:00
readmeFile , err = getReadmeFileFromPath ( ctx . Repo . Commit , entry . GetSubJumpablePathName ( ) )
if err != nil {
ctx . ServerError ( "getReadmeFileFromPath" , err )
2022-04-26 23:31:15 +03:00
return nil , ""
2020-02-22 02:04:20 +03:00
}
if readmeFile != nil {
readmeFile . name = entry . Name ( ) + "/" + readmeFile . name
2021-11-16 21:18:25 +03:00
readmeTreelink = treeLink + "/" + util . PathEscapeSegments ( entry . GetSubJumpablePathName ( ) )
2020-02-22 02:04:20 +03:00
break
}
}
}
2022-04-26 23:31:15 +03:00
return readmeFile , readmeTreelink
}
2020-02-22 02:04:20 +03:00
2022-04-26 23:31:15 +03:00
func renderReadmeFile ( ctx * context . Context , readmeFile * namedBlob , readmeTreelink string ) {
ctx . Data [ "RawFileLink" ] = ""
ctx . Data [ "ReadmeInList" ] = true
ctx . Data [ "ReadmeExist" ] = true
ctx . Data [ "FileIsSymlink" ] = readmeFile . isSymlink
2016-08-30 12:08:38 +03:00
2022-04-26 23:31:15 +03:00
dataRc , err := readmeFile . blob . DataAsync ( )
if err != nil {
ctx . ServerError ( "Data" , err )
return
}
defer dataRc . Close ( )
buf := make ( [ ] byte , 1024 )
n , _ := util . ReadAtMost ( dataRc , buf )
buf = buf [ : n ]
st := typesniffer . DetectContentType ( buf )
isTextFile := st . IsText ( )
ctx . Data [ "FileIsText" ] = isTextFile
ctx . Data [ "FileName" ] = readmeFile . name
fileSize := int64 ( 0 )
isLFSFile := false
ctx . Data [ "IsLFSFile" ] = false
// FIXME: what happens when README file is an image?
if isTextFile && setting . LFS . StartServer {
pointer , _ := lfs . ReadPointerFromBuffer ( buf )
if pointer . IsValid ( ) {
meta , err := models . GetLFSMetaObjectByOid ( ctx . Repo . Repository . ID , pointer . Oid )
if err != nil && err != models . ErrLFSObjectNotExist {
ctx . ServerError ( "GetLFSMetaObject" , err )
return
}
if meta != nil {
ctx . Data [ "IsLFSFile" ] = true
isLFSFile = true
// OK read the lfs object
var err error
dataRc , err = lfs . ReadMetaObject ( pointer )
if err != nil {
ctx . ServerError ( "ReadMetaObject" , err )
2019-02-21 23:57:16 +03:00
return
}
2022-04-26 23:31:15 +03:00
defer dataRc . Close ( )
2019-02-21 23:57:16 +03:00
2022-04-26 23:31:15 +03:00
buf = make ( [ ] byte , 1024 )
n , err = util . ReadAtMost ( dataRc , buf )
if err != nil {
ctx . ServerError ( "Data" , err )
return
}
buf = buf [ : n ]
2019-02-21 23:57:16 +03:00
2022-04-26 23:31:15 +03:00
st = typesniffer . DetectContentType ( buf )
isTextFile = st . IsText ( )
ctx . Data [ "IsTextFile" ] = isTextFile
2019-02-21 23:57:16 +03:00
2022-04-26 23:31:15 +03:00
fileSize = meta . Size
ctx . Data [ "FileSize" ] = meta . Size
filenameBase64 := base64 . RawURLEncoding . EncodeToString ( [ ] byte ( readmeFile . name ) )
ctx . Data [ "RawFileLink" ] = fmt . Sprintf ( "%s.git/info/lfs/objects/%s/%s" , ctx . Repo . Repository . HTMLURL ( ) , url . PathEscape ( meta . Oid ) , url . PathEscape ( filenameBase64 ) )
2019-02-21 23:57:16 +03:00
}
}
2022-04-26 23:31:15 +03:00
}
2019-02-21 23:57:16 +03:00
2022-04-26 23:31:15 +03:00
if ! isTextFile {
return
}
2019-02-21 23:57:16 +03:00
2022-04-26 23:31:15 +03:00
if ! isLFSFile {
fileSize = readmeFile . blob . Size ( )
}
2022-01-07 04:18:52 +03:00
2022-04-26 23:31:15 +03:00
if fileSize >= setting . UI . MaxDisplayFileSize {
// Pretend that this is a normal text file to display 'This file is too large to be shown'
ctx . Data [ "IsFileTooLarge" ] = true
ctx . Data [ "IsTextFile" ] = true
ctx . Data [ "FileSize" ] = fileSize
return
2016-08-30 12:08:38 +03:00
}
2016-08-15 09:02:14 +03:00
2022-04-26 23:31:15 +03:00
rd := charset . ToUTF8WithFallbackReader ( io . MultiReader ( bytes . NewReader ( buf ) , dataRc ) )
if markupType := markup . Type ( readmeFile . name ) ; markupType != "" {
ctx . Data [ "IsMarkup" ] = true
ctx . Data [ "MarkupType" ] = string ( markupType )
var result strings . Builder
err := markup . Render ( & markup . RenderContext {
Ctx : ctx ,
Filename : readmeFile . name ,
URLPrefix : readmeTreelink ,
Metas : ctx . Repo . Repository . ComposeDocumentMetas ( ) ,
GitRepo : ctx . Repo . GitRepo ,
} , rd , & result )
if err != nil {
log . Error ( "Render failed: %v then fallback" , err )
buf := & bytes . Buffer { }
ctx . Data [ "EscapeStatus" ] , _ = charset . EscapeControlReader ( rd , buf )
ctx . Data [ "FileContent" ] = strings . ReplaceAll (
gotemplate . HTMLEscapeString ( buf . String ( ) ) , "\n" , ` <br> ` ,
)
} else {
ctx . Data [ "EscapeStatus" ] , ctx . Data [ "FileContent" ] = charset . EscapeControlString ( result . String ( ) )
}
} else {
ctx . Data [ "IsRenderedHTML" ] = true
buf := & bytes . Buffer { }
ctx . Data [ "EscapeStatus" ] , err = charset . EscapeControlReader ( rd , buf )
if err != nil {
log . Error ( "Read failed: %v" , err )
}
ctx . Data [ "FileContent" ] = strings . ReplaceAll (
gotemplate . HTMLEscapeString ( buf . String ( ) ) , "\n" , ` <br> ` ,
)
2016-08-30 12:08:38 +03:00
}
}
2014-09-30 12:39:53 +04:00
2016-08-30 12:08:38 +03:00
func renderFile ( ctx * context . Context , entry * git . TreeEntry , treeLink , rawLink string ) {
ctx . Data [ "IsViewFile" ] = true
blob := entry . Blob ( )
2017-11-29 04:50:39 +03:00
dataRc , err := blob . DataAsync ( )
2016-08-30 12:08:38 +03:00
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "DataAsync" , err )
2016-08-30 12:08:38 +03:00
return
}
2017-11-29 04:50:39 +03:00
defer dataRc . Close ( )
2014-07-26 08:24:27 +04:00
2021-12-15 10:50:11 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.file.title" , ctx . Repo . Repository . Name + "/" + path . Base ( ctx . Repo . TreePath ) , ctx . Repo . RefName )
2018-09-10 20:01:49 +03:00
2019-02-21 23:57:16 +03:00
fileSize := blob . Size ( )
2020-02-22 02:04:20 +03:00
ctx . Data [ "FileIsSymlink" ] = entry . IsLink ( )
2016-08-30 12:08:38 +03:00
ctx . Data [ "FileName" ] = blob . Name ( )
2021-11-16 21:18:25 +03:00
ctx . Data [ "RawFileLink" ] = rawLink + "/" + util . PathEscapeSegments ( ctx . Repo . TreePath )
2016-08-30 12:08:38 +03:00
buf := make ( [ ] byte , 1024 )
2021-10-25 00:12:43 +03:00
n , _ := util . ReadAtMost ( dataRc , buf )
2016-08-31 23:59:23 +03:00
buf = buf [ : n ]
2016-08-30 12:08:38 +03:00
2021-06-05 15:32:19 +03:00
st := typesniffer . DetectContentType ( buf )
isTextFile := st . IsText ( )
2019-02-12 18:09:43 +03:00
isLFSFile := false
2021-08-11 03:31:13 +03:00
isDisplayingSource := ctx . FormString ( "display" ) == "source"
2021-01-13 06:45:19 +03:00
isDisplayingRendered := ! isDisplayingSource
2022-01-20 20:46:10 +03:00
// Check for LFS meta file
2019-02-21 23:57:16 +03:00
if isTextFile && setting . LFS . StartServer {
2021-04-09 01:25:57 +03:00
pointer , _ := lfs . ReadPointerFromBuffer ( buf )
if pointer . IsValid ( ) {
2021-12-10 04:27:50 +03:00
meta , err := models . GetLFSMetaObjectByOid ( ctx . Repo . Repository . ID , pointer . Oid )
2019-02-21 23:57:16 +03:00
if err != nil && err != models . ErrLFSObjectNotExist {
ctx . ServerError ( "GetLFSMetaObject" , err )
return
}
2021-04-09 01:25:57 +03:00
if meta != nil {
isLFSFile = true
2019-02-21 23:57:16 +03:00
2021-04-09 01:25:57 +03:00
// OK read the lfs object
var err error
dataRc , err = lfs . ReadMetaObject ( pointer )
if err != nil {
ctx . ServerError ( "ReadMetaObject" , err )
return
}
defer dataRc . Close ( )
buf = make ( [ ] byte , 1024 )
2021-10-25 00:12:43 +03:00
n , err = util . ReadAtMost ( dataRc , buf )
if err != nil {
2021-04-09 01:25:57 +03:00
ctx . ServerError ( "Data" , err )
return
}
buf = buf [ : n ]
2019-02-12 18:09:43 +03:00
2021-06-05 15:32:19 +03:00
st = typesniffer . DetectContentType ( buf )
isTextFile = st . IsText ( )
2021-04-09 01:25:57 +03:00
fileSize = meta . Size
2021-11-16 21:18:25 +03:00
ctx . Data [ "RawFileLink" ] = ctx . Repo . RepoLink + "/media/" + ctx . Repo . BranchNameSubURL ( ) + "/" + util . PathEscapeSegments ( ctx . Repo . TreePath )
2021-04-09 01:25:57 +03:00
}
2016-12-26 04:16:37 +03:00
}
}
2021-02-05 04:29:42 +03:00
2021-06-05 15:32:19 +03:00
isRepresentableAsText := st . IsRepresentableAsText ( )
2021-02-05 04:29:42 +03:00
if ! isRepresentableAsText {
// If we can't show plain text, always try to render.
isDisplayingSource = false
isDisplayingRendered = true
}
ctx . Data [ "IsLFSFile" ] = isLFSFile
ctx . Data [ "FileSize" ] = fileSize
ctx . Data [ "IsTextFile" ] = isTextFile
ctx . Data [ "IsRepresentableAsText" ] = isRepresentableAsText
ctx . Data [ "IsDisplayingSource" ] = isDisplayingSource
ctx . Data [ "IsDisplayingRendered" ] = isDisplayingRendered
ctx . Data [ "IsTextSource" ] = isTextFile || isDisplayingSource
2019-10-30 00:32:21 +03:00
// Check LFS Lock
2021-12-10 04:27:50 +03:00
lfsLock , err := models . GetTreePathLock ( ctx . Repo . Repository . ID , ctx . Repo . TreePath )
2019-10-30 00:32:21 +03:00
ctx . Data [ "LFSLock" ] = lfsLock
if err != nil {
ctx . ServerError ( "GetTreePathLock" , err )
return
}
if lfsLock != nil {
2021-11-24 12:49:20 +03:00
u , err := user_model . GetUserByID ( lfsLock . OwnerID )
if err != nil {
ctx . ServerError ( "GetTreePathLock" , err )
return
}
ctx . Data [ "LFSLockOwner" ] = u . DisplayName ( )
2022-02-16 19:22:25 +03:00
ctx . Data [ "LFSLockOwnerHomeLink" ] = u . HomeLink ( )
2019-10-30 00:32:21 +03:00
ctx . Data [ "LFSLockHint" ] = ctx . Tr ( "repo.editor.this_file_locked" )
}
2016-12-26 04:16:37 +03:00
2016-08-30 12:08:38 +03:00
// Assume file is not editable first.
2019-02-12 18:09:43 +03:00
if isLFSFile {
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.cannot_edit_lfs_files" )
2021-01-13 06:45:19 +03:00
} else if ! isRepresentableAsText {
2016-08-30 12:08:38 +03:00
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.cannot_edit_non_text_files" )
}
switch {
2021-01-13 06:45:19 +03:00
case isRepresentableAsText :
2021-06-05 15:32:19 +03:00
if st . IsSvgImage ( ) {
2021-01-13 06:45:19 +03:00
ctx . Data [ "IsImageFile" ] = true
ctx . Data [ "HasSourceRenderedToggle" ] = true
}
2019-02-21 23:57:16 +03:00
if fileSize >= setting . UI . MaxDisplayFileSize {
2016-08-30 12:08:38 +03:00
ctx . Data [ "IsFileTooLarge" ] = true
break
2014-07-26 08:24:27 +04:00
}
2021-04-20 01:25:08 +03:00
rd := charset . ToUTF8WithFallbackReader ( io . MultiReader ( bytes . NewReader ( buf ) , dataRc ) )
2022-04-10 18:01:35 +03:00
shouldRenderSource := ctx . FormString ( "display" ) == "source"
2017-09-21 08:20:14 +03:00
readmeExist := markup . IsReadmeFile ( blob . Name ( ) )
2016-08-30 12:08:38 +03:00
ctx . Data [ "ReadmeExist" ] = readmeExist
2022-04-10 18:01:35 +03:00
markupType := markup . Type ( blob . Name ( ) )
2022-06-09 00:46:39 +03:00
// If the markup is detected by custom markup renderer it should not be reset later on
// to not pass it down to the render context.
detected := false
if markupType == "" {
detected = true
markupType = markup . DetectRendererType ( blob . Name ( ) , bytes . NewReader ( buf ) )
}
2022-04-10 18:01:35 +03:00
if markupType != "" {
ctx . Data [ "HasSourceRenderedToggle" ] = true
}
if markupType != "" && ! shouldRenderSource {
2017-10-17 02:17:22 +03:00
ctx . Data [ "IsMarkup" ] = true
2019-08-16 01:09:50 +03:00
ctx . Data [ "MarkupType" ] = markupType
2021-04-20 01:25:08 +03:00
var result strings . Builder
2022-06-09 00:46:39 +03:00
if ! detected {
markupType = ""
}
2021-04-20 01:25:08 +03:00
err := markup . Render ( & markup . RenderContext {
2021-07-13 00:13:59 +03:00
Ctx : ctx ,
2022-06-09 00:46:39 +03:00
Type : markupType ,
2021-04-20 01:25:08 +03:00
Filename : blob . Name ( ) ,
URLPrefix : path . Dir ( treeLink ) ,
Metas : ctx . Repo . Repository . ComposeDocumentMetas ( ) ,
2021-06-21 01:39:12 +03:00
GitRepo : ctx . Repo . GitRepo ,
2021-04-20 01:25:08 +03:00
} , rd , & result )
if err != nil {
ctx . ServerError ( "Render" , err )
return
}
2022-01-07 04:18:52 +03:00
ctx . Data [ "EscapeStatus" ] , ctx . Data [ "FileContent" ] = charset . EscapeControlString ( result . String ( ) )
2022-04-10 18:01:35 +03:00
} else if readmeExist && ! shouldRenderSource {
2022-01-07 04:18:52 +03:00
buf := & bytes . Buffer { }
2017-10-16 10:04:34 +03:00
ctx . Data [ "IsRenderedHTML" ] = true
2022-01-07 04:18:52 +03:00
ctx . Data [ "EscapeStatus" ] , _ = charset . EscapeControlReader ( rd , buf )
2020-10-11 23:27:20 +03:00
ctx . Data [ "FileContent" ] = strings . ReplaceAll (
2022-01-07 04:18:52 +03:00
gotemplate . HTMLEscapeString ( buf . String ( ) ) , "\n" , ` <br> ` ,
2018-06-10 21:42:16 +03:00
)
2016-08-30 12:08:38 +03:00
} else {
2021-09-22 08:38:34 +03:00
buf , _ := io . ReadAll ( rd )
2020-07-01 00:34:03 +03:00
lineNums := linesBytesCount ( buf )
ctx . Data [ "NumLines" ] = strconv . Itoa ( lineNums )
2019-10-29 19:05:26 +03:00
ctx . Data [ "NumLinesSet" ] = true
2021-11-17 23:37:00 +03:00
language := ""
indexFilename , worktree , deleteTemporaryFile , err := ctx . Repo . GitRepo . ReadTreeToTemporaryIndex ( ctx . Repo . CommitID )
if err == nil {
defer deleteTemporaryFile ( )
filename2attribute2info , err := ctx . Repo . GitRepo . CheckAttribute ( git . CheckAttributeOpts {
CachedOnly : true ,
Attributes : [ ] string { "linguist-language" , "gitlab-language" } ,
Filenames : [ ] string { ctx . Repo . TreePath } ,
IndexFile : indexFilename ,
WorkTree : worktree ,
} )
if err != nil {
log . Error ( "Unable to load attributes for %-v:%s. Error: %v" , ctx . Repo . Repository , ctx . Repo . TreePath , err )
}
language = filename2attribute2info [ ctx . Repo . TreePath ] [ "linguist-language" ]
if language == "" || language == "unspecified" {
language = filename2attribute2info [ ctx . Repo . TreePath ] [ "gitlab-language" ]
}
if language == "unspecified" {
language = ""
}
}
2022-01-07 04:18:52 +03:00
fileContent := highlight . File ( lineNums , blob . Name ( ) , language , buf )
status , _ := charset . EscapeControlReader ( bytes . NewReader ( buf ) , io . Discard )
ctx . Data [ "EscapeStatus" ] = status
statuses := make ( [ ] charset . EscapeStatus , len ( fileContent ) )
for i , line := range fileContent {
statuses [ i ] , fileContent [ i ] = charset . EscapeControlString ( line )
}
ctx . Data [ "FileContent" ] = fileContent
ctx . Data [ "LineEscapeStatus" ] = statuses
2014-09-26 16:55:13 +04:00
}
2019-02-12 18:09:43 +03:00
if ! isLFSFile {
2022-04-28 18:45:33 +03:00
if ctx . Repo . CanEnableEditor ( ctx . Doer ) {
2022-03-22 10:03:22 +03:00
if lfsLock != nil && lfsLock . OwnerID != ctx . Doer . ID {
2019-10-30 00:32:21 +03:00
ctx . Data [ "CanEditFile" ] = false
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.this_file_locked" )
} else {
ctx . Data [ "CanEditFile" ] = true
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.edit_this_file" )
}
2019-02-12 18:09:43 +03:00
} else if ! ctx . Repo . IsViewBranch {
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.must_be_on_a_branch" )
2022-04-28 18:45:33 +03:00
} else if ! ctx . Repo . CanWriteToBranch ( ctx . Doer , ctx . Repo . BranchName ) {
2019-02-12 18:09:43 +03:00
ctx . Data [ "EditFileTooltip" ] = ctx . Tr ( "repo.editor.fork_before_edit" )
}
2016-08-28 11:41:44 +03:00
}
2016-08-30 12:08:38 +03:00
2021-06-05 15:32:19 +03:00
case st . IsPDF ( ) :
2016-08-30 12:08:38 +03:00
ctx . Data [ "IsPDFFile" ] = true
2021-06-05 15:32:19 +03:00
case st . IsVideo ( ) :
2016-12-20 11:09:11 +03:00
ctx . Data [ "IsVideoFile" ] = true
2021-06-05 15:32:19 +03:00
case st . IsAudio ( ) :
2018-10-30 05:17:26 +03:00
ctx . Data [ "IsAudioFile" ] = true
2021-06-05 15:32:19 +03:00
case st . IsImage ( ) && ( setting . UI . SVG . Enabled || ! st . IsSvgImage ( ) ) :
2016-08-30 12:08:38 +03:00
ctx . Data [ "IsImageFile" ] = true
2019-10-21 09:54:18 +03:00
default :
if fileSize >= setting . UI . MaxDisplayFileSize {
ctx . Data [ "IsFileTooLarge" ] = true
break
}
if markupType := markup . Type ( blob . Name ( ) ) ; markupType != "" {
2021-04-20 01:25:08 +03:00
rd := io . MultiReader ( bytes . NewReader ( buf ) , dataRc )
2019-10-21 09:54:18 +03:00
ctx . Data [ "IsMarkup" ] = true
ctx . Data [ "MarkupType" ] = markupType
2021-04-20 01:25:08 +03:00
var result strings . Builder
err := markup . Render ( & markup . RenderContext {
2021-07-13 00:13:59 +03:00
Ctx : ctx ,
2021-04-20 01:25:08 +03:00
Filename : blob . Name ( ) ,
URLPrefix : path . Dir ( treeLink ) ,
Metas : ctx . Repo . Repository . ComposeDocumentMetas ( ) ,
2021-06-21 01:39:12 +03:00
GitRepo : ctx . Repo . GitRepo ,
2021-04-20 01:25:08 +03:00
} , rd , & result )
if err != nil {
ctx . ServerError ( "Render" , err )
return
}
2022-01-07 04:18:52 +03:00
ctx . Data [ "EscapeStatus" ] , ctx . Data [ "FileContent" ] = charset . EscapeControlString ( result . String ( ) )
2019-10-21 09:54:18 +03:00
}
2014-07-26 08:24:27 +04:00
}
2022-04-28 18:45:33 +03:00
if ctx . Repo . CanEnableEditor ( ctx . Doer ) {
2022-03-22 10:03:22 +03:00
if lfsLock != nil && lfsLock . OwnerID != ctx . Doer . ID {
2019-10-30 00:32:21 +03:00
ctx . Data [ "CanDeleteFile" ] = false
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.this_file_locked" )
} else {
ctx . Data [ "CanDeleteFile" ] = true
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.delete_this_file" )
}
2016-08-30 12:08:38 +03:00
} else if ! ctx . Repo . IsViewBranch {
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.must_be_on_a_branch" )
2022-04-28 18:45:33 +03:00
} else if ! ctx . Repo . CanWriteToBranch ( ctx . Doer , ctx . Repo . BranchName ) {
2016-08-30 12:08:38 +03:00
ctx . Data [ "DeleteFileTooltip" ] = ctx . Tr ( "repo.editor.must_have_write_access" )
}
}
2019-10-13 16:23:14 +03:00
func safeURL ( address string ) string {
u , err := url . Parse ( address )
if err != nil {
return address
}
u . User = nil
return u . String ( )
}
2021-10-08 16:08:22 +03:00
func checkHomeCodeViewable ( ctx * context . Context ) {
2018-11-28 14:26:14 +03:00
if len ( ctx . Repo . Units ) > 0 {
2019-10-13 16:23:14 +03:00
if ctx . Repo . Repository . IsBeingCreated ( ) {
task , err := models . GetMigratingTask ( ctx . Repo . Repository . ID )
if err != nil {
2021-11-13 14:28:50 +03:00
if models . IsErrTaskDoesNotExist ( err ) {
ctx . Data [ "Repo" ] = ctx . Repo
ctx . Data [ "CloneAddr" ] = ""
ctx . Data [ "Failed" ] = true
ctx . HTML ( http . StatusOK , tplMigrating )
return
}
2019-10-13 16:23:14 +03:00
ctx . ServerError ( "models.GetMigratingTask" , err )
return
}
cfg , err := task . MigrateConfig ( )
if err != nil {
ctx . ServerError ( "task.MigrateConfig" , err )
return
}
ctx . Data [ "Repo" ] = ctx . Repo
ctx . Data [ "MigrateTask" ] = task
ctx . Data [ "CloneAddr" ] = safeURL ( cfg . CloneAddr )
2021-06-30 22:14:53 +03:00
ctx . Data [ "Failed" ] = task . Status == structs . TaskStatusFailed
2021-04-05 18:30:52 +03:00
ctx . HTML ( http . StatusOK , tplMigrating )
2019-10-13 16:23:14 +03:00
return
}
2021-03-01 03:47:30 +03:00
if ctx . IsSigned {
// Set repo notification-status read if unread
2022-05-20 17:08:52 +03:00
if err := models . SetRepoReadBy ( ctx , ctx . Repo . Repository . ID , ctx . Doer . ID ) ; err != nil {
2021-03-01 03:47:30 +03:00
ctx . ServerError ( "ReadBy" , err )
return
}
}
2021-11-09 22:57:58 +03:00
var firstUnit * unit_model . Unit
2018-11-28 14:26:14 +03:00
for _ , repoUnit := range ctx . Repo . Units {
2021-11-09 22:57:58 +03:00
if repoUnit . Type == unit_model . TypeCode {
2017-10-01 16:50:56 +03:00
return
}
2021-11-09 22:57:58 +03:00
unit , ok := unit_model . Units [ repoUnit . Type ]
2017-10-01 16:50:56 +03:00
if ok && ( firstUnit == nil || ! firstUnit . IsLessThan ( unit ) ) {
firstUnit = & unit
}
2017-05-18 17:54:24 +03:00
}
2017-10-01 16:50:56 +03:00
if firstUnit != nil {
2021-11-16 21:18:25 +03:00
ctx . Redirect ( fmt . Sprintf ( "%s%s" , ctx . Repo . Repository . Link ( ) , firstUnit . URI ) )
2017-05-18 17:54:24 +03:00
return
}
}
2018-01-11 00:34:17 +03:00
ctx . NotFound ( "Home" , fmt . Errorf ( ctx . Tr ( "units.error.no_unit_allowed_repo" ) ) )
2017-05-18 17:54:24 +03:00
}
2021-10-08 16:08:22 +03:00
// Home render repository home page
func Home ( ctx * context . Context ) {
2022-03-13 19:40:47 +03:00
isFeed , _ , showFeedType := feed . GetFeedType ( ctx . Params ( ":reponame" ) , ctx . Req )
if isFeed {
feed . ShowRepoFeed ( ctx , ctx . Repo . Repository , showFeedType )
return
}
ctx . Data [ "FeedURL" ] = ctx . Repo . Repository . HTMLURL ( )
2021-10-08 16:08:22 +03:00
checkHomeCodeViewable ( ctx )
if ctx . Written ( ) {
return
}
renderCode ( ctx )
}
// LastCommit returns lastCommit data for the provided branch/tag/commit and directory (in url) and filenames in body
func LastCommit ( ctx * context . Context ) {
checkHomeCodeViewable ( ctx )
if ctx . Written ( ) {
return
}
renderDirectoryFiles ( ctx , 0 )
if ctx . Written ( ) {
return
}
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 ) - 2 ]
}
}
branchLink := ctx . Repo . RepoLink + "/src/" + ctx . Repo . BranchNameSubURL ( )
ctx . Data [ "BranchLink" ] = branchLink
ctx . HTML ( http . StatusOK , tplRepoViewList )
}
func renderDirectoryFiles ( ctx * context . Context , timeout time . Duration ) git . Entries {
tree , err := ctx . Repo . Commit . SubTree ( ctx . Repo . TreePath )
if err != nil {
ctx . NotFoundOrServerError ( "Repo.Commit.SubTree" , git . IsErrNotExist , err )
return nil
}
2021-11-16 21:18:25 +03:00
ctx . Data [ "LastCommitLoaderURL" ] = ctx . Repo . RepoLink + "/lastcommit/" + url . PathEscape ( ctx . Repo . CommitID ) + "/" + util . PathEscapeSegments ( ctx . Repo . TreePath )
2021-10-08 16:08:22 +03:00
// 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 nil
}
if ! entry . IsDir ( ) {
ctx . NotFoundOrServerError ( "Repo.Commit.GetTreeEntryByPath" , git . IsErrNotExist , err )
return nil
}
allEntries , err := tree . ListEntries ( )
if err != nil {
ctx . ServerError ( "ListEntries" , err )
return nil
}
allEntries . CustomSort ( base . NaturalSortLess )
commitInfoCtx := gocontext . Context ( ctx )
if timeout > 0 {
var cancel gocontext . CancelFunc
commitInfoCtx , cancel = gocontext . WithTimeout ( ctx , timeout )
defer cancel ( )
}
var c * git . LastCommitCache
if setting . CacheService . LastCommit . Enabled && ctx . Repo . CommitsCount >= setting . CacheService . LastCommit . CommitsCount {
c = git . NewLastCommitCache ( ctx . Repo . Repository . FullName ( ) , ctx . Repo . GitRepo , setting . LastCommitCacheTTLSeconds , cache . GetCache ( ) )
}
selected := map [ string ] bool { }
for _ , pth := range ctx . FormStrings ( "f[]" ) {
selected [ pth ] = true
}
entries := allEntries
if len ( selected ) > 0 {
entries = make ( git . Entries , 0 , len ( selected ) )
for _ , entry := range allEntries {
if selected [ entry . Name ( ) ] {
entries = append ( entries , entry )
}
}
}
var latestCommit * git . Commit
ctx . Data [ "Files" ] , latestCommit , err = entries . GetCommitsInfo ( commitInfoCtx , ctx . Repo . Commit , ctx . Repo . TreePath , c )
if err != nil {
ctx . ServerError ( "GetCommitsInfo" , err )
return nil
}
// Show latest commit info of repository in table header,
// or of directory if not in root directory.
ctx . Data [ "LatestCommit" ] = latestCommit
if latestCommit != nil {
2021-12-10 11:14:24 +03:00
verification := asymkey_model . ParseCommitWithSignature ( latestCommit )
2021-10-08 16:08:22 +03:00
2021-12-10 11:14:24 +03:00
if err := asymkey_model . CalculateTrustStatus ( verification , ctx . Repo . Repository . GetTrustModel ( ) , func ( user * user_model . User ) ( bool , error ) {
2022-02-02 13:10:06 +03:00
return models . IsOwnerMemberCollaborator ( ctx . Repo . Repository , user . ID )
2021-12-10 11:14:24 +03:00
} , nil ) ; err != nil {
2021-10-08 16:08:22 +03:00
ctx . ServerError ( "CalculateTrustStatus" , err )
return nil
}
ctx . Data [ "LatestCommitVerification" ] = verification
2021-11-24 12:49:20 +03:00
ctx . Data [ "LatestCommitUser" ] = user_model . ValidateCommitWithEmail ( latestCommit )
2021-10-08 16:08:22 +03:00
}
2022-05-20 17:08:52 +03:00
statuses , _ , err := models . GetLatestCommitStatus ( ctx , ctx . Repo . Repository . ID , ctx . Repo . Commit . ID . String ( ) , db . ListOptions { } )
2021-10-08 16:08:22 +03:00
if err != nil {
log . Error ( "GetLatestCommitStatus: %v" , err )
}
ctx . Data [ "LatestCommitStatus" ] = models . CalcCommitStatus ( statuses )
ctx . Data [ "LatestCommitStatuses" ] = statuses
branchLink := ctx . Repo . RepoLink + "/src/" + ctx . Repo . BranchNameSubURL ( )
treeLink := branchLink
if len ( ctx . Repo . TreePath ) > 0 {
2021-11-16 21:18:25 +03:00
treeLink += "/" + util . PathEscapeSegments ( ctx . Repo . TreePath )
2021-10-08 16:08:22 +03:00
}
ctx . Data [ "TreeLink" ] = treeLink
ctx . Data [ "SSHDomain" ] = setting . SSH . Domain
return allEntries
}
2020-02-11 12:34:17 +03:00
func renderLanguageStats ( ctx * context . Context ) {
2021-12-10 04:27:50 +03:00
langs , err := repo_model . GetTopLanguageStats ( ctx . Repo . Repository , 5 )
2020-02-11 12:34:17 +03:00
if err != nil {
ctx . ServerError ( "Repo.GetTopLanguageStats" , err )
return
}
ctx . Data [ "LanguageStats" ] = langs
}
2020-05-06 00:51:49 +03:00
func renderRepoTopics ( ctx * context . Context ) {
2021-12-12 18:48:20 +03:00
topics , _ , err := repo_model . FindTopics ( & repo_model . FindTopicOptions {
2020-05-06 00:51:49 +03:00
RepoID : ctx . Repo . Repository . ID ,
} )
if err != nil {
ctx . ServerError ( "models.FindTopics" , err )
return
}
ctx . Data [ "Topics" ] = topics
}
2017-05-18 17:54:24 +03:00
func renderCode ( ctx * context . Context ) {
2017-03-18 13:59:07 +03:00
ctx . Data [ "PageIsViewCode" ] = true
2019-01-18 03:01:04 +03:00
if ctx . Repo . Repository . IsEmpty {
2021-12-24 18:36:26 +03:00
reallyEmpty , err := ctx . Repo . GitRepo . IsEmpty ( )
if err != nil {
ctx . ServerError ( "GitRepo.IsEmpty" , err )
return
}
if reallyEmpty {
ctx . HTML ( http . StatusOK , tplRepoEMPTY )
return
}
// the repo is not really empty, so we should update the modal in database
// such problem may be caused by:
// 1) an error occurs during pushing/receiving. 2) the user replaces an empty git repo manually
// and even more: the IsEmpty flag is deeply broken and should be removed with the UI changed to manage to cope with empty repos.
// it's possible for a repository to be non-empty by that flag but still 500
// because there are no branches - only tags -or the default branch is non-extant as it has been 0-pushed.
ctx . Repo . Repository . IsEmpty = false
2022-05-20 17:08:52 +03:00
if err = repo_model . UpdateRepositoryCols ( ctx , ctx . Repo . Repository , "is_empty" ) ; err != nil {
2021-12-24 18:36:26 +03:00
ctx . ServerError ( "UpdateRepositoryCols" , err )
return
}
2022-06-06 11:01:49 +03:00
if err = repo_module . UpdateRepoSize ( ctx , ctx . Repo . Repository ) ; err != nil {
2021-12-24 18:36:26 +03:00
ctx . ServerError ( "UpdateRepoSize" , err )
return
}
2017-03-18 13:59:07 +03:00
}
2016-08-30 12:08:38 +03:00
title := ctx . Repo . Repository . Owner . Name + "/" + ctx . Repo . Repository . Name
if len ( ctx . Repo . Repository . Description ) > 0 {
title += ": " + ctx . Repo . Repository . Description
}
ctx . Data [ "Title" ] = title
2017-10-30 05:04:25 +03:00
branchLink := ctx . Repo . RepoLink + "/src/" + ctx . Repo . BranchNameSubURL ( )
2016-08-30 12:08:38 +03:00
treeLink := branchLink
2017-10-30 05:04:25 +03:00
rawLink := ctx . Repo . RepoLink + "/raw/" + ctx . Repo . BranchNameSubURL ( )
2016-08-30 12:08:38 +03:00
if len ( ctx . Repo . TreePath ) > 0 {
2021-11-16 21:18:25 +03:00
treeLink += "/" + util . PathEscapeSegments ( ctx . Repo . TreePath )
2016-08-30 12:08:38 +03:00
}
2018-04-11 05:51:44 +03:00
// Get Topics of this repo
2020-05-06 00:51:49 +03:00
renderRepoTopics ( ctx )
if ctx . Written ( ) {
2018-04-11 05:51:44 +03:00
return
}
2016-08-30 12:08:38 +03:00
// 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
}
2020-02-11 12:34:17 +03:00
renderLanguageStats ( ctx )
if ctx . Written ( ) {
return
}
2016-08-30 12:08:38 +03:00
if entry . IsDir ( ) {
renderDirectory ( ctx , treeLink )
} else {
renderFile ( ctx , entry , treeLink , rawLink )
}
if ctx . Written ( ) {
return
}
2014-07-26 08:24:27 +04:00
2016-08-30 12:08:38 +03:00
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 ] , "/" ) )
2014-07-26 08:24:27 +04:00
}
ctx . Data [ "HasParentPath" ] = true
2016-08-09 22:56:00 +03:00
if len ( paths ) - 2 >= 0 {
ctx . Data [ "ParentPath" ] = "/" + paths [ len ( paths ) - 2 ]
2014-07-26 08:24:27 +04:00
}
}
2016-08-09 22:56:00 +03:00
ctx . Data [ "Paths" ] = paths
2016-08-28 01:25:01 +03:00
ctx . Data [ "TreeLink" ] = treeLink
2016-08-30 12:08:38 +03:00
ctx . Data [ "TreeNames" ] = treeNames
2014-07-26 08:24:27 +04:00
ctx . Data [ "BranchLink" ] = branchLink
2021-04-05 18:30:52 +03:00
ctx . HTML ( http . StatusOK , tplRepoHome )
2014-07-26 08:24:27 +04:00
}
2015-11-17 07:28:46 +03:00
2021-07-08 14:38:13 +03:00
// RenderUserCards render a page show users according the input template
2021-11-24 12:49:20 +03:00
func RenderUserCards ( ctx * context . Context , total int , getter func ( opts db . ListOptions ) ( [ ] * user_model . User , error ) , tpl base . TplName ) {
2021-07-29 04:42:15 +03:00
page := ctx . FormInt ( "page" )
2015-11-17 07:28:46 +03:00
if page <= 0 {
page = 1
}
2019-04-20 07:15:19 +03:00
pager := context . NewPagination ( total , models . ItemsPerPage , page , 5 )
2015-11-17 07:28:46 +03:00
ctx . Data [ "Page" ] = pager
2021-09-24 14:32:56 +03:00
items , err := getter ( db . ListOptions {
2021-02-04 20:23:46 +03:00
Page : pager . Paginater . Current ( ) ,
PageSize : models . ItemsPerPage ,
} )
2015-11-17 07:28:46 +03:00
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "getter" , err )
2015-11-17 07:28:46 +03:00
return
}
2015-12-21 15:24:11 +03:00
ctx . Data [ "Cards" ] = items
2015-11-17 07:28:46 +03:00
2021-04-05 18:30:52 +03:00
ctx . HTML ( http . StatusOK , tpl )
2015-11-17 07:28:46 +03:00
}
2016-11-21 13:03:29 +03:00
// Watchers render repository's watch users
2016-03-11 19:56:52 +03:00
func Watchers ( ctx * context . Context ) {
2015-11-17 07:28:46 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.watchers" )
2015-12-21 15:24:11 +03:00
ctx . Data [ "CardsTitle" ] = ctx . Tr ( "repo.watchers" )
2015-11-17 07:28:46 +03:00
ctx . Data [ "PageIsWatchers" ] = true
2019-04-20 07:15:19 +03:00
2021-12-10 04:27:50 +03:00
RenderUserCards ( ctx , ctx . Repo . Repository . NumWatches , func ( opts db . ListOptions ) ( [ ] * user_model . User , error ) {
2021-12-12 18:48:20 +03:00
return repo_model . GetRepoWatchers ( ctx . Repo . Repository . ID , opts )
2021-12-10 04:27:50 +03:00
} , tplWatchers )
2015-11-17 07:28:46 +03:00
}
2016-11-21 13:03:29 +03:00
// Stars render repository's starred users
2016-03-11 19:56:52 +03:00
func Stars ( ctx * context . Context ) {
2015-11-17 07:28:46 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.stargazers" )
2015-12-21 15:24:11 +03:00
ctx . Data [ "CardsTitle" ] = ctx . Tr ( "repo.stargazers" )
2015-11-17 07:28:46 +03:00
ctx . Data [ "PageIsStargazers" ] = true
2021-11-24 12:49:20 +03:00
RenderUserCards ( ctx , ctx . Repo . Repository . NumStars , func ( opts db . ListOptions ) ( [ ] * user_model . User , error ) {
2021-12-12 18:48:20 +03:00
return repo_model . GetStargazers ( ctx . Repo . Repository , opts )
2021-11-22 18:21:55 +03:00
} , tplWatchers )
2015-11-17 07:28:46 +03:00
}
2015-11-17 07:33:40 +03:00
2016-11-21 13:03:29 +03:00
// Forks render repository's forked users
2016-03-11 19:56:52 +03:00
func Forks ( ctx * context . Context ) {
2015-11-17 07:33:40 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repos.forks" )
2021-11-18 17:45:56 +03:00
page := ctx . FormInt ( "page" )
if page <= 0 {
page = 1
}
pager := context . NewPagination ( ctx . Repo . Repository . NumForks , models . ItemsPerPage , page , 5 )
ctx . Data [ "Page" ] = pager
2021-12-12 18:48:20 +03:00
forks , err := repo_model . GetForks ( ctx . Repo . Repository , db . ListOptions {
2021-11-18 17:45:56 +03:00
Page : pager . Paginater . Current ( ) ,
PageSize : models . ItemsPerPage ,
} )
2015-11-17 07:33:40 +03:00
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "GetForks" , err )
2015-11-17 07:33:40 +03:00
return
}
for _ , fork := range forks {
2022-03-22 18:22:54 +03:00
if err = fork . GetOwner ( ctx ) ; err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "GetOwner" , err )
2015-11-17 07:33:40 +03:00
return
}
}
2021-11-18 17:45:56 +03:00
2015-11-17 07:33:40 +03:00
ctx . Data [ "Forks" ] = forks
2021-04-05 18:30:52 +03:00
ctx . HTML ( http . StatusOK , tplForks )
2015-11-17 07:33:40 +03:00
}