mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08:00 
			
		
		
		
	Add activity feeds API (#23494)
Close #5666 Add APIs for getting activity feeds.
This commit is contained in:
		@@ -754,6 +754,8 @@ func Routes(ctx gocontext.Context) *web.Route {
 | 
			
		||||
						Post(bind(api.CreateAccessTokenOption{}), user.CreateAccessToken)
 | 
			
		||||
					m.Combo("/{id}").Delete(user.DeleteAccessToken)
 | 
			
		||||
				}, reqBasicAuth())
 | 
			
		||||
 | 
			
		||||
				m.Get("/activities/feeds", user.ListUserActivityFeeds)
 | 
			
		||||
			}, context_service.UserAssignmentAPI())
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
@@ -1177,6 +1179,7 @@ func Routes(ctx gocontext.Context) *web.Route {
 | 
			
		||||
				m.Get("/issue_config", context.ReferencesGitRepo(), repo.GetIssueConfig)
 | 
			
		||||
				m.Get("/issue_config/validate", context.ReferencesGitRepo(), repo.ValidateIssueConfig)
 | 
			
		||||
				m.Get("/languages", reqRepoReader(unit.TypeCode), repo.GetLanguages)
 | 
			
		||||
				m.Get("/activities/feeds", repo.ListRepoActivityFeeds)
 | 
			
		||||
			}, repoAssignment())
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
@@ -1234,6 +1237,7 @@ func Routes(ctx gocontext.Context) *web.Route {
 | 
			
		||||
					Patch(bind(api.EditHookOption{}), org.EditHook).
 | 
			
		||||
					Delete(org.DeleteHook)
 | 
			
		||||
			}, reqToken(auth_model.AccessTokenScopeAdminOrgHook), reqOrgOwnership(), reqWebhooksEnabled())
 | 
			
		||||
			m.Get("/activities/feeds", org.ListOrgActivityFeeds)
 | 
			
		||||
		}, orgAssignment(true))
 | 
			
		||||
		m.Group("/teams/{teamid}", func() {
 | 
			
		||||
			m.Combo("").Get(reqToken(auth_model.AccessTokenScopeReadOrg), org.GetTeam).
 | 
			
		||||
@@ -1253,6 +1257,7 @@ func Routes(ctx gocontext.Context) *web.Route {
 | 
			
		||||
					Delete(reqToken(auth_model.AccessTokenScopeWriteOrg), org.RemoveTeamRepository).
 | 
			
		||||
					Get(reqToken(auth_model.AccessTokenScopeReadOrg), org.GetTeamRepo)
 | 
			
		||||
			})
 | 
			
		||||
			m.Get("/activities/feeds", org.ListTeamActivityFeeds)
 | 
			
		||||
		}, orgAssignment(false, true), reqToken(""), reqTeamMembership())
 | 
			
		||||
 | 
			
		||||
		m.Group("/admin", func() {
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,7 @@ package org
 | 
			
		||||
import (
 | 
			
		||||
	"net/http"
 | 
			
		||||
 | 
			
		||||
	activities_model "code.gitea.io/gitea/models/activities"
 | 
			
		||||
	"code.gitea.io/gitea/models/db"
 | 
			
		||||
	"code.gitea.io/gitea/models/organization"
 | 
			
		||||
	"code.gitea.io/gitea/models/perm"
 | 
			
		||||
@@ -370,3 +371,69 @@ func Delete(ctx *context.APIContext) {
 | 
			
		||||
	}
 | 
			
		||||
	ctx.Status(http.StatusNoContent)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func ListOrgActivityFeeds(ctx *context.APIContext) {
 | 
			
		||||
	// swagger:operation GET /orgs/{org}/activities/feeds organization orgListActivityFeeds
 | 
			
		||||
	// ---
 | 
			
		||||
	// summary: List an organization's activity feeds
 | 
			
		||||
	// produces:
 | 
			
		||||
	// - application/json
 | 
			
		||||
	// parameters:
 | 
			
		||||
	// - name: org
 | 
			
		||||
	//   in: path
 | 
			
		||||
	//   description: name of the org
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   required: true
 | 
			
		||||
	// - name: date
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: the date of the activities to be found
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   format: date
 | 
			
		||||
	// - name: page
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page number of results to return (1-based)
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// - name: limit
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page size of results
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// responses:
 | 
			
		||||
	//   "200":
 | 
			
		||||
	//     "$ref": "#/responses/ActivityFeedsList"
 | 
			
		||||
	//   "404":
 | 
			
		||||
	//     "$ref": "#/responses/notFound"
 | 
			
		||||
 | 
			
		||||
	includePrivate := false
 | 
			
		||||
	if ctx.IsSigned {
 | 
			
		||||
		if ctx.Doer.IsAdmin {
 | 
			
		||||
			includePrivate = true
 | 
			
		||||
		} else {
 | 
			
		||||
			org := organization.OrgFromUser(ctx.ContextUser)
 | 
			
		||||
			isMember, err := org.IsOrgMember(ctx.Doer.ID)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				ctx.Error(http.StatusInternalServerError, "IsOrgMember", err)
 | 
			
		||||
				return
 | 
			
		||||
			}
 | 
			
		||||
			includePrivate = isMember
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	listOptions := utils.GetListOptions(ctx)
 | 
			
		||||
 | 
			
		||||
	opts := activities_model.GetFeedsOptions{
 | 
			
		||||
		RequestedUser:  ctx.ContextUser,
 | 
			
		||||
		Actor:          ctx.Doer,
 | 
			
		||||
		IncludePrivate: includePrivate,
 | 
			
		||||
		Date:           ctx.FormString("date"),
 | 
			
		||||
		ListOptions:    listOptions,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	feeds, count, err := activities_model.GetFeeds(ctx, opts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		ctx.Error(http.StatusInternalServerError, "GetFeeds", err)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	ctx.SetTotalCountHeader(count)
 | 
			
		||||
 | 
			
		||||
	ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -9,6 +9,7 @@ import (
 | 
			
		||||
	"net/http"
 | 
			
		||||
 | 
			
		||||
	"code.gitea.io/gitea/models"
 | 
			
		||||
	activities_model "code.gitea.io/gitea/models/activities"
 | 
			
		||||
	"code.gitea.io/gitea/models/organization"
 | 
			
		||||
	"code.gitea.io/gitea/models/perm"
 | 
			
		||||
	access_model "code.gitea.io/gitea/models/perm/access"
 | 
			
		||||
@@ -792,3 +793,55 @@ func SearchTeam(ctx *context.APIContext) {
 | 
			
		||||
		"data": apiTeams,
 | 
			
		||||
	})
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func ListTeamActivityFeeds(ctx *context.APIContext) {
 | 
			
		||||
	// swagger:operation GET /teams/{id}/activities/feeds organization orgListTeamActivityFeeds
 | 
			
		||||
	// ---
 | 
			
		||||
	// summary: List a team's activity feeds
 | 
			
		||||
	// produces:
 | 
			
		||||
	// - application/json
 | 
			
		||||
	// parameters:
 | 
			
		||||
	// - name: id
 | 
			
		||||
	//   in: path
 | 
			
		||||
	//   description: id of the team
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	//   format: int64
 | 
			
		||||
	//   required: true
 | 
			
		||||
	// - name: date
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: the date of the activities to be found
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   format: date
 | 
			
		||||
	// - name: page
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page number of results to return (1-based)
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// - name: limit
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page size of results
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// responses:
 | 
			
		||||
	//   "200":
 | 
			
		||||
	//     "$ref": "#/responses/ActivityFeedsList"
 | 
			
		||||
	//   "404":
 | 
			
		||||
	//     "$ref": "#/responses/notFound"
 | 
			
		||||
 | 
			
		||||
	listOptions := utils.GetListOptions(ctx)
 | 
			
		||||
 | 
			
		||||
	opts := activities_model.GetFeedsOptions{
 | 
			
		||||
		RequestedTeam:  ctx.Org.Team,
 | 
			
		||||
		Actor:          ctx.Doer,
 | 
			
		||||
		IncludePrivate: true,
 | 
			
		||||
		Date:           ctx.FormString("date"),
 | 
			
		||||
		ListOptions:    listOptions,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	feeds, count, err := activities_model.GetFeeds(ctx, opts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		ctx.Error(http.StatusInternalServerError, "GetFeeds", err)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	ctx.SetTotalCountHeader(count)
 | 
			
		||||
 | 
			
		||||
	ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -10,6 +10,7 @@ import (
 | 
			
		||||
	"strings"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	activities_model "code.gitea.io/gitea/models/activities"
 | 
			
		||||
	"code.gitea.io/gitea/models/db"
 | 
			
		||||
	"code.gitea.io/gitea/models/organization"
 | 
			
		||||
	"code.gitea.io/gitea/models/perm"
 | 
			
		||||
@@ -1199,3 +1200,59 @@ func ValidateIssueConfig(ctx *context.APIContext) {
 | 
			
		||||
		ctx.JSON(http.StatusOK, api.IssueConfigValidation{Valid: false, Message: err.Error()})
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func ListRepoActivityFeeds(ctx *context.APIContext) {
 | 
			
		||||
	// swagger:operation GET /repos/{owner}/{repo}/activities/feeds repository repoListActivityFeeds
 | 
			
		||||
	// ---
 | 
			
		||||
	// summary: List a repository's activity feeds
 | 
			
		||||
	// produces:
 | 
			
		||||
	// - application/json
 | 
			
		||||
	// parameters:
 | 
			
		||||
	// - name: owner
 | 
			
		||||
	//   in: path
 | 
			
		||||
	//   description: owner of the repo
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   required: true
 | 
			
		||||
	// - name: repo
 | 
			
		||||
	//   in: path
 | 
			
		||||
	//   description: name of the repo
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   required: true
 | 
			
		||||
	// - name: date
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: the date of the activities to be found
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   format: date
 | 
			
		||||
	// - name: page
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page number of results to return (1-based)
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// - name: limit
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page size of results
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// responses:
 | 
			
		||||
	//   "200":
 | 
			
		||||
	//     "$ref": "#/responses/ActivityFeedsList"
 | 
			
		||||
	//   "404":
 | 
			
		||||
	//     "$ref": "#/responses/notFound"
 | 
			
		||||
 | 
			
		||||
	listOptions := utils.GetListOptions(ctx)
 | 
			
		||||
 | 
			
		||||
	opts := activities_model.GetFeedsOptions{
 | 
			
		||||
		RequestedRepo:  ctx.Repo.Repository,
 | 
			
		||||
		Actor:          ctx.Doer,
 | 
			
		||||
		IncludePrivate: true,
 | 
			
		||||
		Date:           ctx.FormString("date"),
 | 
			
		||||
		ListOptions:    listOptions,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	feeds, count, err := activities_model.GetFeeds(ctx, opts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		ctx.Error(http.StatusInternalServerError, "GetFeeds", err)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	ctx.SetTotalCountHeader(count)
 | 
			
		||||
 | 
			
		||||
	ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										15
									
								
								routers/api/v1/swagger/activity.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								routers/api/v1/swagger/activity.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
// Copyright 2023 The Gitea Authors. All rights reserved.
 | 
			
		||||
// SPDX-License-Identifier: MIT
 | 
			
		||||
 | 
			
		||||
package swagger
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	api "code.gitea.io/gitea/modules/structs"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// ActivityFeedsList
 | 
			
		||||
// swagger:response ActivityFeedsList
 | 
			
		||||
type swaggerActivityFeedsList struct {
 | 
			
		||||
	// in:body
 | 
			
		||||
	Body []api.Activity `json:"body"`
 | 
			
		||||
}
 | 
			
		||||
@@ -145,3 +145,60 @@ func GetUserHeatmapData(ctx *context.APIContext) {
 | 
			
		||||
	}
 | 
			
		||||
	ctx.JSON(http.StatusOK, heatmap)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func ListUserActivityFeeds(ctx *context.APIContext) {
 | 
			
		||||
	// swagger:operation GET /users/{username}/activities/feeds user userListActivityFeeds
 | 
			
		||||
	// ---
 | 
			
		||||
	// summary: List a user's activity feeds
 | 
			
		||||
	// produces:
 | 
			
		||||
	// - application/json
 | 
			
		||||
	// parameters:
 | 
			
		||||
	// - name: username
 | 
			
		||||
	//   in: path
 | 
			
		||||
	//   description: username of user
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   required: true
 | 
			
		||||
	// - name: only-performed-by
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: if true, only show actions performed by the requested user
 | 
			
		||||
	//   type: boolean
 | 
			
		||||
	// - name: date
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: the date of the activities to be found
 | 
			
		||||
	//   type: string
 | 
			
		||||
	//   format: date
 | 
			
		||||
	// - name: page
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page number of results to return (1-based)
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// - name: limit
 | 
			
		||||
	//   in: query
 | 
			
		||||
	//   description: page size of results
 | 
			
		||||
	//   type: integer
 | 
			
		||||
	// responses:
 | 
			
		||||
	//   "200":
 | 
			
		||||
	//     "$ref": "#/responses/ActivityFeedsList"
 | 
			
		||||
	//   "404":
 | 
			
		||||
	//     "$ref": "#/responses/notFound"
 | 
			
		||||
 | 
			
		||||
	includePrivate := ctx.IsSigned && (ctx.Doer.IsAdmin || ctx.Doer.ID == ctx.ContextUser.ID)
 | 
			
		||||
	listOptions := utils.GetListOptions(ctx)
 | 
			
		||||
 | 
			
		||||
	opts := activities_model.GetFeedsOptions{
 | 
			
		||||
		RequestedUser:   ctx.ContextUser,
 | 
			
		||||
		Actor:           ctx.Doer,
 | 
			
		||||
		IncludePrivate:  includePrivate,
 | 
			
		||||
		OnlyPerformedBy: ctx.FormBool("only-performed-by"),
 | 
			
		||||
		Date:            ctx.FormString("date"),
 | 
			
		||||
		ListOptions:     listOptions,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	feeds, count, err := activities_model.GetFeeds(ctx, opts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		ctx.Error(http.StatusInternalServerError, "GetFeeds", err)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	ctx.SetTotalCountHeader(count)
 | 
			
		||||
 | 
			
		||||
	ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user