mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 11:33:11 +00:00 
			
		
		
		
	Old activities are shown like before, new commits are displayed like commits in e.g. the commits list. _(Second commit)_ | New signed commits | Old (signed) commits | |:--:|:--:| |  |  | Additionally the sha box was moved in an own component to unify the usage. _(First commit)_ Closes #1824 <!--start release-notes-assistant--> ## Release notes <!--URL:https://codeberg.org/forgejo/forgejo--> - User Interface features - [PR](https://codeberg.org/forgejo/forgejo/pulls/6933): <!--number 6933 --><!--line 0 --><!--description U2hvdyBpZiBjb21taXQgaXMgdmVyaWZpZWQgaW4gYWN0aXZpdHkgZmVlZCBvZiBhbiB1c2VyIG9yIGFuIG9yZ2FuaXphdGlvbiBmb3IgbmV3IGFjdGl2aXR5-->Show if commit is verified in activity feed of an user or an organization for new activity<!--description--> <!--end release-notes-assistant--> Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/6933 Reviewed-by: Gusted <gusted@noreply.codeberg.org> Co-authored-by: Beowulf <beowulf@beocode.eu> Co-committed-by: Beowulf <beowulf@beocode.eu>
		
			
				
	
	
		
			351 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			351 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2021 The Gitea Authors. All rights reserved.
 | |
| // Copyright 2025 The Forgejo Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package feed
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"html"
 | |
| 	"html/template"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	activities_model "forgejo.org/models/activities"
 | |
| 	issues_model "forgejo.org/models/issues"
 | |
| 	repo_model "forgejo.org/models/repo"
 | |
| 	"forgejo.org/modules/log"
 | |
| 	"forgejo.org/modules/markup"
 | |
| 	"forgejo.org/modules/markup/markdown"
 | |
| 	"forgejo.org/modules/setting"
 | |
| 	"forgejo.org/modules/templates"
 | |
| 	"forgejo.org/modules/util"
 | |
| 	"forgejo.org/services/context"
 | |
| 
 | |
| 	"github.com/gorilla/feeds"
 | |
| 	"github.com/jaytaylor/html2text"
 | |
| )
 | |
| 
 | |
| func toBranchLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/src/branch/" + util.PathEscapeSegments(act.GetBranch())
 | |
| }
 | |
| 
 | |
| func toTagLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/src/tag/" + util.PathEscapeSegments(act.GetTag())
 | |
| }
 | |
| 
 | |
| func toIssueLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/issues/" + url.PathEscape(act.GetIssueInfos()[0])
 | |
| }
 | |
| 
 | |
| func toPullLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/pulls/" + url.PathEscape(act.GetIssueInfos()[0])
 | |
| }
 | |
| 
 | |
| func toSrcLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/src/" + util.PathEscapeSegments(act.GetBranch())
 | |
| }
 | |
| 
 | |
| func toReleaseLink(ctx *context.Context, act *activities_model.Action) string {
 | |
| 	return act.GetRepoAbsoluteLink(ctx) + "/releases/tag/" + util.PathEscapeSegments(act.GetBranch())
 | |
| }
 | |
| 
 | |
| // renderMarkdown creates a minimal markdown render context from an action.
 | |
| // If rendering fails, the original markdown text is returned
 | |
| func renderMarkdown(ctx *context.Context, act *activities_model.Action, content string) template.HTML {
 | |
| 	markdownCtx := &markup.RenderContext{
 | |
| 		Ctx: ctx,
 | |
| 		Links: markup.Links{
 | |
| 			Base: act.GetRepoLink(ctx),
 | |
| 		},
 | |
| 		Type: markdown.MarkupName,
 | |
| 		Metas: map[string]string{
 | |
| 			"user": act.GetRepoUserName(ctx),
 | |
| 			"repo": act.GetRepoName(ctx),
 | |
| 		},
 | |
| 	}
 | |
| 	markdown, err := markdown.RenderString(markdownCtx, content)
 | |
| 	if err != nil {
 | |
| 		return templates.SanitizeHTML(content) // old code did so: use SanitizeHTML to render in tmpl
 | |
| 	}
 | |
| 	return markdown
 | |
| }
 | |
| 
 | |
| // feedActionsToFeedItems convert gitea's Action feed to feeds Item
 | |
| func feedActionsToFeedItems(ctx *context.Context, actions activities_model.ActionList) (items []*feeds.Item, err error) {
 | |
| 	for _, act := range actions {
 | |
| 		act.LoadActUser(ctx)
 | |
| 
 | |
| 		// TODO: the code seems quite strange (maybe not right)
 | |
| 		// sometimes it uses text content but sometimes it uses HTML content
 | |
| 		// it should clearly defines which kind of content it should use for the feed items: plan text or rich HTML
 | |
| 		var title, desc string
 | |
| 		var content template.HTML
 | |
| 
 | |
| 		link := &feeds.Link{Href: act.GetCommentHTMLURL(ctx)}
 | |
| 
 | |
| 		// title
 | |
| 		title = act.ActUser.GetDisplayName() + " "
 | |
| 		var titleExtra template.HTML
 | |
| 		switch act.OpType {
 | |
| 		case activities_model.ActionCreateRepo:
 | |
| 			titleExtra = ctx.Locale.Tr("action.create_repo", act.GetRepoAbsoluteLink(ctx), act.ShortRepoPath(ctx))
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 		case activities_model.ActionRenameRepo:
 | |
| 			titleExtra = ctx.Locale.Tr("action.rename_repo", act.GetContent(), act.GetRepoAbsoluteLink(ctx), act.ShortRepoPath(ctx))
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 		case activities_model.ActionCommitRepo:
 | |
| 			link.Href = toBranchLink(ctx, act)
 | |
| 			if len(act.Content) != 0 {
 | |
| 				titleExtra = ctx.Locale.Tr("action.commit_repo", act.GetRepoAbsoluteLink(ctx), link.Href, act.GetBranch(), act.ShortRepoPath(ctx))
 | |
| 			} else {
 | |
| 				titleExtra = ctx.Locale.Tr("action.create_branch", act.GetRepoAbsoluteLink(ctx), link.Href, act.GetBranch(), act.ShortRepoPath(ctx))
 | |
| 			}
 | |
| 		case activities_model.ActionCreateIssue:
 | |
| 			link.Href = toIssueLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.create_issue", link.Href, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionCreatePullRequest:
 | |
| 			link.Href = toPullLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.create_pull_request", link.Href, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionTransferRepo:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.transfer_repo", act.GetContent(), act.GetRepoAbsoluteLink(ctx), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionPushTag:
 | |
| 			link.Href = toTagLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.push_tag", act.GetRepoAbsoluteLink(ctx), link.Href, act.GetTag(), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionCommentIssue:
 | |
| 			issueLink := toIssueLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = issueLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.comment_issue", issueLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionMergePullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = pullLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.merge_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionAutoMergePullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = pullLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.auto_merge_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionCloseIssue:
 | |
| 			issueLink := toIssueLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = issueLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.close_issue", issueLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionReopenIssue:
 | |
| 			issueLink := toIssueLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = issueLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.reopen_issue", issueLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionClosePullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = pullLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.close_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionReopenPullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = pullLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.reopen_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionDeleteTag:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.delete_tag", act.GetRepoAbsoluteLink(ctx), act.GetTag(), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionDeleteBranch:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.delete_branch", act.GetRepoAbsoluteLink(ctx), html.EscapeString(act.GetBranch()), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionMirrorSyncPush:
 | |
| 			srcLink := toSrcLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = srcLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.mirror_sync_push", act.GetRepoAbsoluteLink(ctx), srcLink, act.GetBranch(), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionMirrorSyncCreate:
 | |
| 			srcLink := toSrcLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = srcLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.mirror_sync_create", act.GetRepoAbsoluteLink(ctx), srcLink, act.GetBranch(), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionMirrorSyncDelete:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.mirror_sync_delete", act.GetRepoAbsoluteLink(ctx), act.GetBranch(), act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionApprovePullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.approve_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionRejectPullRequest:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.reject_pull_request", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionCommentPull:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.comment_pull", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx))
 | |
| 		case activities_model.ActionPublishRelease:
 | |
| 			releaseLink := toReleaseLink(ctx, act)
 | |
| 			if link.Href == "#" {
 | |
| 				link.Href = releaseLink
 | |
| 			}
 | |
| 			titleExtra = ctx.Locale.Tr("action.publish_release", act.GetRepoAbsoluteLink(ctx), releaseLink, act.ShortRepoPath(ctx), act.Content)
 | |
| 		case activities_model.ActionPullReviewDismissed:
 | |
| 			pullLink := toPullLink(ctx, act)
 | |
| 			titleExtra = ctx.Locale.Tr("action.review_dismissed", pullLink, act.GetIssueInfos()[0], act.ShortRepoPath(ctx), act.GetIssueInfos()[1])
 | |
| 		case activities_model.ActionStarRepo:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.starred_repo", act.GetRepoAbsoluteLink(ctx), act.GetRepoPath(ctx))
 | |
| 		case activities_model.ActionWatchRepo:
 | |
| 			link.Href = act.GetRepoAbsoluteLink(ctx)
 | |
| 			titleExtra = ctx.Locale.Tr("action.watched_repo", act.GetRepoAbsoluteLink(ctx), act.GetRepoPath(ctx))
 | |
| 		default:
 | |
| 			return nil, fmt.Errorf("unknown action type: %v", act.OpType)
 | |
| 		}
 | |
| 
 | |
| 		// description & content
 | |
| 		{
 | |
| 			switch act.OpType {
 | |
| 			case activities_model.ActionCommitRepo, activities_model.ActionMirrorSyncPush:
 | |
| 				push := templates.ActionContent2Commits(ctx, act)
 | |
| 
 | |
| 				for _, commit := range push.Commits {
 | |
| 					if len(desc) != 0 {
 | |
| 						desc += "\n\n"
 | |
| 					}
 | |
| 					desc += fmt.Sprintf("<a href=\"%s\">%s</a>\n%s",
 | |
| 						html.EscapeString(fmt.Sprintf("%s/commit/%s", act.GetRepoAbsoluteLink(ctx), commit.Sha1)),
 | |
| 						commit.Sha1,
 | |
| 						templates.RenderCommitMessage(ctx, commit.Message, nil),
 | |
| 					)
 | |
| 				}
 | |
| 
 | |
| 				if push.Len > 1 {
 | |
| 					link = &feeds.Link{Href: fmt.Sprintf("%s/%s", setting.AppSubURL, push.CompareURL)}
 | |
| 				} else if push.Len == 1 {
 | |
| 					link = &feeds.Link{Href: fmt.Sprintf("%s/commit/%s", act.GetRepoAbsoluteLink(ctx), push.Commits[0].Sha1)}
 | |
| 				}
 | |
| 
 | |
| 			case activities_model.ActionCreateIssue, activities_model.ActionCreatePullRequest:
 | |
| 				desc = strings.Join(act.GetIssueInfos(), "#")
 | |
| 				content = renderMarkdown(ctx, act, act.GetIssueContent(ctx))
 | |
| 			case activities_model.ActionCommentIssue, activities_model.ActionApprovePullRequest, activities_model.ActionRejectPullRequest, activities_model.ActionCommentPull:
 | |
| 				desc = act.GetIssueTitle(ctx)
 | |
| 				comment := act.GetIssueInfos()[1]
 | |
| 				if strings.HasSuffix(comment, "…") {
 | |
| 					// Comment was truncated get the full content from the database.
 | |
| 					// This truncation is done in `NotifyCreateIssueComment`.
 | |
| 					commentModel, err := issues_model.GetCommentByID(ctx, act.CommentID)
 | |
| 					if err != nil {
 | |
| 						log.Error("Couldn't get comment[%d] for RSS feed: %v", act.CommentID, err)
 | |
| 					} else {
 | |
| 						comment = commentModel.Content
 | |
| 					}
 | |
| 				}
 | |
| 				if len(comment) != 0 {
 | |
| 					desc += "\n\n" + string(renderMarkdown(ctx, act, comment))
 | |
| 				}
 | |
| 			case activities_model.ActionMergePullRequest, activities_model.ActionAutoMergePullRequest:
 | |
| 				desc = act.GetIssueInfos()[1]
 | |
| 			case activities_model.ActionCloseIssue, activities_model.ActionReopenIssue, activities_model.ActionClosePullRequest, activities_model.ActionReopenPullRequest:
 | |
| 				desc = act.GetIssueTitle(ctx)
 | |
| 			case activities_model.ActionPullReviewDismissed:
 | |
| 				desc = ctx.Locale.TrString("action.review_dismissed_reason") + "\n\n" + act.GetIssueInfos()[2]
 | |
| 			}
 | |
| 		}
 | |
| 		if len(content) == 0 {
 | |
| 			content = templates.SanitizeHTML(desc)
 | |
| 		}
 | |
| 
 | |
| 		// It's a common practice for feed generators to use plain text titles.
 | |
| 		// See https://codeberg.org/forgejo/forgejo/pulls/1595
 | |
| 		plainTitle, err := html2text.FromString(title+" "+string(titleExtra), html2text.Options{OmitLinks: true})
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 
 | |
| 		items = append(items, &feeds.Item{
 | |
| 			Title:       plainTitle,
 | |
| 			Link:        link,
 | |
| 			Description: desc,
 | |
| 			IsPermaLink: "false",
 | |
| 			Author: &feeds.Author{
 | |
| 				Name:  act.ActUser.GetDisplayName(),
 | |
| 				Email: act.ActUser.GetEmail(),
 | |
| 			},
 | |
| 			Id:      fmt.Sprintf("%v: %v", strconv.FormatInt(act.ID, 10), link.Href),
 | |
| 			Created: act.CreatedUnix.AsTime(),
 | |
| 			Content: string(content),
 | |
| 		})
 | |
| 	}
 | |
| 	return items, err
 | |
| }
 | |
| 
 | |
| // GetFeedType return if it is a feed request and altered name and feed type.
 | |
| func GetFeedType(name string, req *http.Request) (bool, string, string) {
 | |
| 	if strings.HasSuffix(name, ".rss") ||
 | |
| 		strings.Contains(req.Header.Get("Accept"), "application/rss+xml") {
 | |
| 		return true, strings.TrimSuffix(name, ".rss"), "rss"
 | |
| 	}
 | |
| 
 | |
| 	if strings.HasSuffix(name, ".atom") ||
 | |
| 		strings.Contains(req.Header.Get("Accept"), "application/atom+xml") {
 | |
| 		return true, strings.TrimSuffix(name, ".atom"), "atom"
 | |
| 	}
 | |
| 
 | |
| 	return false, name, ""
 | |
| }
 | |
| 
 | |
| // feedActionsToFeedItems convert repository releases into feed items.
 | |
| func releasesToFeedItems(ctx *context.Context, releases repo_model.ReleaseList) (items []*feeds.Item, err error) {
 | |
| 	if err := releases.LoadAttributes(ctx); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	composeCache := make(map[int64]map[string]string)
 | |
| 	for _, rel := range releases {
 | |
| 		var title string
 | |
| 		var content template.HTML
 | |
| 
 | |
| 		if rel.IsTag {
 | |
| 			title = rel.TagName
 | |
| 		} else {
 | |
| 			title = rel.Title
 | |
| 		}
 | |
| 
 | |
| 		metas, ok := composeCache[rel.RepoID]
 | |
| 		if !ok {
 | |
| 			metas = rel.Repo.ComposeMetas(ctx)
 | |
| 			composeCache[rel.RepoID] = metas
 | |
| 		}
 | |
| 
 | |
| 		link := &feeds.Link{Href: rel.HTMLURL()}
 | |
| 		content, err = markdown.RenderString(&markup.RenderContext{
 | |
| 			Ctx: ctx,
 | |
| 			Links: markup.Links{
 | |
| 				Base: rel.Repo.Link(),
 | |
| 			},
 | |
| 			Metas: metas,
 | |
| 		}, rel.Note)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 
 | |
| 		items = append(items, &feeds.Item{
 | |
| 			Title:   title,
 | |
| 			Link:    link,
 | |
| 			Created: rel.CreatedUnix.AsTime(),
 | |
| 			Author: &feeds.Author{
 | |
| 				Name:  rel.Publisher.GetDisplayName(),
 | |
| 				Email: rel.Publisher.GetEmail(),
 | |
| 			},
 | |
| 			Id:      fmt.Sprintf("%v: %v", strconv.FormatInt(rel.ID, 10), link.Href),
 | |
| 			Content: string(content),
 | |
| 		})
 | |
| 	}
 | |
| 
 | |
| 	return items, err
 | |
| }
 |