mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 17:08:25 +00:00 
			
		
		
		
	close #33086 * Add a special value for "SSH_USER" setting: `(DOER_USERNAME)` * Improve parseRepositoryURL and add tests (now it doesn't have hard dependency on some setting values) Many changes are just adding "ctx" and "doer" argument to functions. By the way, improve app.example.ini, remove all `%(key)s` syntax, it only makes messy and no user really cares about it. Document: https://gitea.com/gitea/docs/pulls/138
		
			
				
	
	
		
			194 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2023 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package templates
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"html/template"
 | |
| 	"mime"
 | |
| 	"path/filepath"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	activities_model "code.gitea.io/gitea/models/activities"
 | |
| 	repo_model "code.gitea.io/gitea/models/repo"
 | |
| 	"code.gitea.io/gitea/modules/git"
 | |
| 	giturl "code.gitea.io/gitea/modules/git/url"
 | |
| 	"code.gitea.io/gitea/modules/json"
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 	"code.gitea.io/gitea/modules/repository"
 | |
| 	"code.gitea.io/gitea/modules/svg"
 | |
| 
 | |
| 	"github.com/editorconfig/editorconfig-core-go/v2"
 | |
| )
 | |
| 
 | |
| func sortArrow(normSort, revSort, urlSort string, isDefault bool) template.HTML {
 | |
| 	// if needed
 | |
| 	if len(normSort) == 0 || len(urlSort) == 0 {
 | |
| 		return ""
 | |
| 	}
 | |
| 
 | |
| 	if len(urlSort) == 0 && isDefault {
 | |
| 		// if sort is sorted as default add arrow tho this table header
 | |
| 		if isDefault {
 | |
| 			return svg.RenderHTML("octicon-triangle-down", 16)
 | |
| 		}
 | |
| 	} else {
 | |
| 		// if sort arg is in url test if it correlates with column header sort arguments
 | |
| 		// the direction of the arrow should indicate the "current sort order", up means ASC(normal), down means DESC(rev)
 | |
| 		if urlSort == normSort {
 | |
| 			// the table is sorted with this header normal
 | |
| 			return svg.RenderHTML("octicon-triangle-up", 16)
 | |
| 		} else if urlSort == revSort {
 | |
| 			// the table is sorted with this header reverse
 | |
| 			return svg.RenderHTML("octicon-triangle-down", 16)
 | |
| 		}
 | |
| 	}
 | |
| 	// the table is NOT sorted with this header
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| // isMultilineCommitMessage checks to see if a commit message contains multiple lines.
 | |
| func isMultilineCommitMessage(msg string) bool {
 | |
| 	return strings.Count(strings.TrimSpace(msg), "\n") >= 1
 | |
| }
 | |
| 
 | |
| // Actioner describes an action
 | |
| type Actioner interface {
 | |
| 	GetOpType() activities_model.ActionType
 | |
| 	GetActUserName(ctx context.Context) string
 | |
| 	GetRepoUserName(ctx context.Context) string
 | |
| 	GetRepoName(ctx context.Context) string
 | |
| 	GetRepoPath(ctx context.Context) string
 | |
| 	GetRepoLink(ctx context.Context) string
 | |
| 	GetBranch() string
 | |
| 	GetContent() string
 | |
| 	GetCreate() time.Time
 | |
| 	GetIssueInfos() []string
 | |
| }
 | |
| 
 | |
| // actionIcon accepts an action operation type and returns an icon class name.
 | |
| func actionIcon(opType activities_model.ActionType) string {
 | |
| 	switch opType {
 | |
| 	case activities_model.ActionCreateRepo, activities_model.ActionTransferRepo, activities_model.ActionRenameRepo:
 | |
| 		return "repo"
 | |
| 	case activities_model.ActionCommitRepo:
 | |
| 		return "git-commit"
 | |
| 	case activities_model.ActionDeleteBranch:
 | |
| 		return "git-branch"
 | |
| 	case activities_model.ActionMergePullRequest, activities_model.ActionAutoMergePullRequest:
 | |
| 		return "git-merge"
 | |
| 	case activities_model.ActionCreatePullRequest:
 | |
| 		return "git-pull-request"
 | |
| 	case activities_model.ActionClosePullRequest:
 | |
| 		return "git-pull-request-closed"
 | |
| 	case activities_model.ActionCreateIssue:
 | |
| 		return "issue-opened"
 | |
| 	case activities_model.ActionCloseIssue:
 | |
| 		return "issue-closed"
 | |
| 	case activities_model.ActionReopenIssue, activities_model.ActionReopenPullRequest:
 | |
| 		return "issue-reopened"
 | |
| 	case activities_model.ActionCommentIssue, activities_model.ActionCommentPull:
 | |
| 		return "comment-discussion"
 | |
| 	case activities_model.ActionMirrorSyncPush, activities_model.ActionMirrorSyncCreate, activities_model.ActionMirrorSyncDelete:
 | |
| 		return "mirror"
 | |
| 	case activities_model.ActionApprovePullRequest:
 | |
| 		return "check"
 | |
| 	case activities_model.ActionRejectPullRequest:
 | |
| 		return "file-diff"
 | |
| 	case activities_model.ActionPublishRelease, activities_model.ActionPushTag, activities_model.ActionDeleteTag:
 | |
| 		return "tag"
 | |
| 	case activities_model.ActionPullReviewDismissed:
 | |
| 		return "x"
 | |
| 	default:
 | |
| 		return "question"
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // ActionContent2Commits converts action content to push commits
 | |
| func ActionContent2Commits(act Actioner) *repository.PushCommits {
 | |
| 	push := repository.NewPushCommits()
 | |
| 
 | |
| 	if act == nil || act.GetContent() == "" {
 | |
| 		return push
 | |
| 	}
 | |
| 
 | |
| 	if err := json.Unmarshal([]byte(act.GetContent()), push); err != nil {
 | |
| 		log.Error("json.Unmarshal:\n%s\nERROR: %v", act.GetContent(), err)
 | |
| 	}
 | |
| 
 | |
| 	if push.Len == 0 {
 | |
| 		push.Len = len(push.Commits)
 | |
| 	}
 | |
| 
 | |
| 	return push
 | |
| }
 | |
| 
 | |
| // migrationIcon returns a SVG name matching the service an issue/comment was migrated from
 | |
| func migrationIcon(hostname string) string {
 | |
| 	switch hostname {
 | |
| 	case "github.com":
 | |
| 		return "octicon-mark-github"
 | |
| 	default:
 | |
| 		return "gitea-git"
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type remoteAddress struct {
 | |
| 	Address  string
 | |
| 	Username string
 | |
| 	Password string
 | |
| }
 | |
| 
 | |
| func mirrorRemoteAddress(ctx context.Context, m *repo_model.Repository, remoteName string) remoteAddress {
 | |
| 	ret := remoteAddress{}
 | |
| 	remoteURL, err := git.GetRemoteAddress(ctx, m.RepoPath(), remoteName)
 | |
| 	if err != nil {
 | |
| 		log.Error("GetRemoteURL %v", err)
 | |
| 		return ret
 | |
| 	}
 | |
| 
 | |
| 	u, err := giturl.ParseGitURL(remoteURL)
 | |
| 	if err != nil {
 | |
| 		log.Error("giturl.Parse %v", err)
 | |
| 		return ret
 | |
| 	}
 | |
| 
 | |
| 	if u.Scheme != "ssh" && u.Scheme != "file" {
 | |
| 		if u.User != nil {
 | |
| 			ret.Username = u.User.Username()
 | |
| 			ret.Password, _ = u.User.Password()
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// The URL stored in the git repo could contain authentication,
 | |
| 	// erase it, or it will be shown in the UI.
 | |
| 	u.User = nil
 | |
| 	ret.Address = u.String()
 | |
| 	// Why not use m.OriginalURL to set ret.Address?
 | |
| 	// It should be OK to use it, since m.OriginalURL should be the same as the authentication-erased URL from the Git repository.
 | |
| 	// However, the old code has already stored authentication in m.OriginalURL when updating mirror settings.
 | |
| 	// That means we need to use "giturl.Parse" for m.OriginalURL again to ensure authentication is erased.
 | |
| 	// Instead of doing this, why not directly use the authentication-erased URL from the Git repository?
 | |
| 	// It should be the same as long as there are no bugs.
 | |
| 
 | |
| 	return ret
 | |
| }
 | |
| 
 | |
| func filenameIsImage(filename string) bool {
 | |
| 	mimeType := mime.TypeByExtension(filepath.Ext(filename))
 | |
| 	return strings.HasPrefix(mimeType, "image/")
 | |
| }
 | |
| 
 | |
| func tabSizeClass(ec *editorconfig.Editorconfig, filename string) string {
 | |
| 	if ec != nil {
 | |
| 		def, err := ec.GetDefinitionForFilename(filename)
 | |
| 		if err == nil && def.TabWidth >= 1 && def.TabWidth <= 16 {
 | |
| 			return "tab-size-" + strconv.Itoa(def.TabWidth)
 | |
| 		}
 | |
| 	}
 | |
| 	return "tab-size-4"
 | |
| }
 |